(function () {
    console.log("window.WebVideoCtrl");
    if (window.WebVideoCtrl) {
        return
    }
    var WebVideoCtrl = function () {
        var m_szWidth = "100%";
        var m_szHeight = "100%";
        var m_options = {
            szversion: "V3.3.0 build20230314",
            szContainerID: "",
            szColorProperty: "",
            szBasePath: "",
            iWndowType: 1,
            bWndFull: true,
            iPackageType: 2,
            bDebugMode: true,
            cbSelWnd: null,
            cbDoubleClickWnd: null,
            cbEvent: null,
            cbInitPluginComplete: null
        };
        var m_pluginOBJECT = null;
        var m_iSelWnd = 0;
        var m_bFullScreen = false;
        var m_deviceSet = [];
        var m_wndSet = [];
        var m_ISAPIProtocol = null;
        var m_utilsInc = null;
        var m_webVideoCtrl = this;
        var m_oLocalCfg = null;
        var PROTOCOL_DEVICE_ISAPI = 1;
        var ERROR_CODE_UNKNOWN = 1e3;
        var ERROR_CODE_NETWORKERROR = 1001;
        var ERROR_CODE_PARAMERROR = 1002;
        var ERROR_CODE_LOGIN_NOLOGIN = 2e3;
        var ERROR_CODE_LOGIN_REPEATLOGIN = 2001;
        var ERROR_CODE_LOGIN_NOSUPPORT = 2002;
        var ERROR_CODE_PLAY_PLUGININITFAIL = 3e3;
        var ERROR_CODE_PLAY_NOREPEATPLAY = 3001;
        var ERROR_CODE_PLAY_PLAYBACKABNORMAL = 3002;
        var ERROR_CODE_PLAY_PLAYBACKSTOP = 3003;
        var ERROR_CODE_PLAY_NOFREESPACE = 3004;
        var ERROR_CODE_TALK_FAIL = 5e3;
        var HTTP_STATUS_OK_200 = 200;
        var HTTP_STATUS_ERROR_403 = 403;
        var PLAY_STATUS_STOP = 0;
        var PLAY_STATUS_REALPLAY = 1;
        var PLAY_STATUS_PLAYBACK = 2;
        var PLAY_STATUS_PAUSE = 3;
        var PLAY_STATUS_FRAME = 4;
        var PLAY_STATUS_REVERSE_PLAYBACK = 5;
        var PLAY_STATUS_REVERSE_PAUSE = 6;
        var PROTOCOLTYPE_PLAY_TCP = 0;
        var PROTOCOLTYPE_PLAY_UDP = 1;
        var DEVICE_TYPE_IPCAMERA = "IPCamera";
        var DEVICE_TYPE_IPDOME = "IPDome";
        var DEVICE_TYPE_IPZOOM = "IPZoom";
        var DEVICE_TYPE_GATEWAY = "Gateway";
        var m_szVersion = "<?xml version='1.0' encoding='utf-8'?><FileVersion>" + "<Platform name='win32'>" + "<localServiceControl>1.0.0.40</localServiceControl>";
        "</Platform>" + "</FileVersion>";
        var _onGetSelectWndInfo = function (iWnd) {
            m_iSelWnd = iWnd;
            if (m_options.cbSelWnd) {
                var arrXml = [];
                arrXml.push("<RealPlayInfo>");
                arrXml.push("<SelectWnd>" + m_iSelWnd + "</SelectWnd>");
                arrXml.push("</RealPlayInfo>");
                m_options.cbSelWnd(m_utilsInc.loadXML(arrXml.join("")))
            }
        };
        var _onMouseEvent = function (oData) {
            if (m_options.cbDoubleClickWnd && 2 === oData.eventType) {
                if (m_options.bWndFull) {
                    var iIndex = m_webVideoCtrl.findWndIndexByIndex(oData.wndIndex);
                    if (iIndex != -1) {
                        m_bFullScreen = !m_bFullScreen
                    }
                }
                m_options.cbDoubleClickWnd(oData.wndIndex, m_bFullScreen)
            }
        };
        var _onPluginEventHandler = function (iWndIndex, iErrorCode, oError) {
            var iNewError = ERROR_CODE_UNKNOWN;
            if (0 === iErrorCode) {
                iNewError = ERROR_CODE_PLAY_PLAYBACKABNORMAL
            } else if (2 === iErrorCode) {
                iNewError = ERROR_CODE_PLAY_PLAYBACKSTOP
            } else if (3 === iErrorCode) {
                iNewError = ERROR_CODE_TALK_FAIL
            } else if (21 === iErrorCode) {
                iNewError = ERROR_CODE_PLAY_NOFREESPACE
            }
            if (ERROR_CODE_PLAY_PLAYBACKABNORMAL == iNewError || ERROR_CODE_PLAY_PLAYBACKSTOP == iNewError) {
                m_webVideoCtrl.I_Stop(iWndIndex)
            } else if (ERROR_CODE_PLAY_NOFREESPACE == iNewError) {
                m_webVideoCtrl.I_StopRecord(iWndIndex)
            } else if (ERROR_CODE_TALK_FAIL == iNewError) {
                m_webVideoCtrl.I_StopVoiceTalk()
            } else {
            }
            if (m_options.cbEvent) {
                m_options.cbEvent(iNewError, iWndIndex, oError)
            }
        };
        var _onKeyBoardEvent = function (iKeyCode) {
            if (100 === parseInt(iKeyCode, 10)) {
                m_bFullScreen = false;
                if (m_options.cbDoubleClickWnd) {
                    m_options.cbDoubleClickWnd(m_iSelWnd, m_bFullScreen)
                }
            }
        };
        var _onZoomInfoCallback = function (oPoints) {
            var iIndex = m_webVideoCtrl.findWndIndexByIndex(m_iSelWnd);
            if (iIndex != -1) {
                var oWndInfo = m_wndSet[iIndex];
                iIndex = m_webVideoCtrl.findDeviceIndexByIP(oWndInfo.szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.set3DZoom(oDeviceInfo, oWndInfo, oPoints, {})
                }
            }
        };
        var _oNoLoginError = {errorCode: ERROR_CODE_LOGIN_NOLOGIN, errorMsg: "The device is not login."};
        var _oUnKnownError = {errorCode: ERROR_CODE_UNKNOWN, errorMsg: "Unknown error."};
        var _oParamsError = {errorCode: ERROR_CODE_PARAMERROR, errorMsg: "Params error."};
        var _printString = function () {
            if (m_options.bDebugMode) {
                var printString = m_utilsInc.formatString(arguments);
                console.log(printString)
            }
        };
        var _initLocalCfg = function () {
            let oPromise = new Promise(function (resolve, reject) {
                m_pluginOBJECT.JS_GetLocalConfig().then(oLocalCofing => {
                    m_oLocalCfg = oLocalCofing;
                    resolve()
                }, () => {
                    reject()
                })
            });
            return oPromise
        };
        var _initDeviceInfo = function (oDeviceInfo) {
            let oPromise = new Promise(function (resolve, reject) {
                let oP1 = oDeviceInfo.oProtocolInc.getDeviceInfo(oDeviceInfo, {});
                let oP2 = oDeviceInfo.oProtocolInc.getAnalogChannelInfo(oDeviceInfo, {});
                let oP3 = oDeviceInfo.oProtocolInc.getAudioInfo(oDeviceInfo, {});
                let oP4 = _getPort(oDeviceInfo);
                let oP5 = oDeviceInfo.oProtocolInc.getDeviceMinusLocalTime(oDeviceInfo);
                Promise.all([oP1, oP2, oP3, oP4, oP5]).then(() => {
                    resolve()
                }, () => {
                    resolve()
                })
            });
            return oPromise
        };
        var _initPlugin = function (szContainerID) {
            let oPromise = new Promise(function (resolve, reject) {
                if (!m_utilsInc.isUndefined(szContainerID)) {
                    m_options.szContainerID = szContainerID
                }
                if (document.getElementById(m_options.szContainerID) == null) {
                    reject(_oParamsError);
                    return
                }
                var oParam = {
                    szId: szContainerID,
                    iType: 1,
                    iWidth: m_szWidth,
                    iHeight: m_szHeight,
                    iMaxSplit: 4,
                    iCurrentSplit: m_options.iWndowType,
                    iServicePortStart: 34686,
                    iServicePortEnd: 34690,
                    oSessionInfo: {
                        sessionID: "11c12b3257f037bb50052db3ac5e342572c3d963622baca122755c482ce8823a",
                        user: "admin",
                        challenge: "275816f02ec2dca22b6a6ae87c7cb7e3",
                        iterations: 100,
                        random: "34765058"
                    },
                    iPluginType: 2,
                    onConnectSuccess: () => {
                        var oElem = $("#" + szContainerID);
                        m_pluginOBJECT.JS_Resize(oElem.width(), oElem.height());
                        if (2 !== m_pluginOBJECT.iPluginMode) {
                            reject({errorCode: ERROR_CODE_PLAY_PLUGININITFAIL, errorMsg: "Plugin init failed."});
                            return
                        }
                        var iWndFull = m_options.bWndFull ? 1 : 0;
                        m_pluginOBJECT.JS_SetFullScreenCapability(iWndFull);
                        m_pluginOBJECT.JS_SetPackageType(m_options.iPackageType);
                        _initPluginEvent();
                        _initLocalCfg().then(() => {
                            resolve()
                        })
                    },
                    onConnectError: () => {
                        reject({errorCode: ERROR_CODE_PLAY_PLUGININITFAIL, errorMsg: "Plugin init failed."})
                    },
                    szBasePath: m_utilsInc.getDirName()
                };
                m_pluginOBJECT = new JSVideoPlugin(oParam);
                window.m_pluginOBJECT = m_pluginOBJECT;
            });
            return oPromise
        };
        var _initPluginEvent = function () {
            m_pluginOBJECT.JS_SetWindowControlCallback({
                onGetSelectWndInfo: iwnd => {
                    _onGetSelectWndInfo(iwnd)
                }, onPluginEventHandler: (iWndIndex, iEventType, iParam2) => {
                    _onPluginEventHandler(iWndIndex, iEventType, iParam2)
                }, KeyBoardEvent: szXml => {
                    _onKeyBoardEvent(szXml)
                }, onMouseEvent: function (oData) {
                    _onMouseEvent(oData)
                }
            })
        };
        var _getPort = function (oDeviceInfo) {
            let oPromise = new Promise(async (resolve, reject) => {
                var oPort = null;
                let bPPPoE = await _getPPPoEEnable(oDeviceInfo);
                if (bPPPoE) {
                    oPort = await _getInternalPort(oDeviceInfo)
                } else {
                    var ipset = await _getDeviceIPAddr(oDeviceInfo);
                    var bSame = false;
                    for (var i = 0; i < ipset.length; i++) {
                        if (ipset[i].ipv4 == oDeviceInfo.szIP || ipset[i].ipv6 == oDeviceInfo.szIP) {
                            bSame = true;
                            break
                        }
                    }
                    if (bSame) {
                        oPort = await _getInternalPort(oDeviceInfo)
                    } else {
                        oPort = await _getExternalPort(oDeviceInfo);
                        if (-1 == oPort.iRtspPort && -1 == oPort.iDevicePort) {
                            oPort = await _getInternalPort(oDeviceInfo)
                        }
                    }
                }
                oDeviceInfo.iRtspPort = oPort.iRtspPort;
                oDeviceInfo.iHttpPort = oPort.iHttpPort;
                resolve(oPort)
            });
            return oPromise
        };
        var _getInternalPort = function (oDeviceInfo) {
            let oPromise = new Promise((resolve, reject) => {
                var iRtspPort = -1, iHttpPort = -1, iDevicePort = -1;
                oDeviceInfo.oProtocolInc.getPortInfo(oDeviceInfo, {
                    async: false, success: function (xmlDoc) {
                        var nodeList = NS.$XML(xmlDoc).find("AdminAccessProtocol", true);
                        iRtspPort = 554;
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            if (NS.$XML(nodeList).eq(i).find("protocol").eq(0).text().toLowerCase() === "rtsp") {
                                iRtspPort = parseInt(NS.$XML(nodeList).eq(i).find("portNo").eq(0).text(), 10)
                            }
                            if (NS.$XML(nodeList).eq(i).find("protocol").eq(0).text().toLowerCase() === "http") {
                                iHttpPort = parseInt(NS.$XML(nodeList).eq(i).find("portNo").eq(0).text(), 10)
                            }
                            if (NS.$XML(nodeList).eq(i).find("protocol").eq(0).text().toLowerCase() === "dev_manage") {
                                iDevicePort = parseInt(NS.$XML(nodeList).eq(i).find("portNo").eq(0).text(), 10)
                            }
                        }
                        resolve({iRtspPort: iRtspPort, iHttpPort: iHttpPort, iDevicePort: iDevicePort})
                    }, error: function () {
                        resolve({iRtspPort: -1, iHttpPort: -1, iDevicePort: -1})
                    }
                })
            });
            return oPromise
        };
        var _getExternalPort = function (oDeviceInfo) {
            let oPromise = new Promise((resolve, reject) => {
                var iRtspPort = -1, iHttpPort = -1, iDevicePort = -1;
                oDeviceInfo.oProtocolInc.getUPnPPortStatus(oDeviceInfo, {
                    async: false, success: function (xmlDoc) {
                        var nodeList = NS.$XML(xmlDoc).find("portStatus", true);
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            if (NS.$XML(nodeList).eq(i).find("internalPort").eq(0).text().toLowerCase() == "rtsp") {
                                iRtspPort = parseInt(NS.$XML(nodeList).eq(i).find("externalPort").eq(0).text(), 10)
                            }
                            if (NS.$XML(nodeList).eq(i).find("internalPort").eq(0).text().toLowerCase() == "http") {
                                iHttpPort = parseInt(NS.$XML(nodeList).eq(i).find("externalPort").eq(0).text(), 10)
                            }
                            if (NS.$XML(nodeList).eq(i).find("internalPort").eq(0).text().toLowerCase() == "admin") {
                                iDevicePort = parseInt(NS.$XML(nodeList).eq(i).find("externalPort").eq(0).text(), 10)
                            }
                        }
                        resolve({iRtspPort: iRtspPort, iHttpPort: iHttpPort, iDevicePort: iDevicePort})
                    }, error: function () {
                        resolve({iRtspPort: -1, iHttpPort: -1, iDevicePort: -1})
                    }
                })
            });
            return oPromise
        };
        var _getDeviceIPAddr = function (oDeviceInfo) {
            let oPromise = new Promise(function (resolve) {
                var arrIP = [];
                oDeviceInfo.oProtocolInc.getNetworkBond(oDeviceInfo, {
                    async: false, success: function (xmlDoc) {
                        if (NS.$XML(xmlDoc).find("enabled").eq(0).text() == "true") {
                            arrIP.push({
                                ipv4: NS.$XML(xmlDoc).find("ipAddress").eq(0).text(),
                                ipv6: NS.$XML(xmlDoc).find("ipv6Address").eq(0).text()
                            });
                            resolve(arrIP)
                        } else {
                            oDeviceInfo.oProtocolInc.getNetworkInterface(oDeviceInfo, {
                                async: false,
                                success: function (xmlDoc) {
                                    var nodeList = NS.$XML(xmlDoc).find("NetworkInterface", true);
                                    for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                                        arrIP.push({
                                            ipv4: NS.$XML(xmlDoc).find("ipAddress").eq(0).text(),
                                            ipv6: NS.$XML(xmlDoc).find("ipv6Address").eq(0).text()
                                        });
                                        break
                                    }
                                    resolve(arrIP)
                                },
                                error: function () {
                                    resolve(arrIP)
                                }
                            })
                        }
                    }, error: function () {
                        oDeviceInfo.oProtocolInc.getNetworkInterface(oDeviceInfo, {
                            async: false,
                            success: function (xmlDoc) {
                                var nodeList = NS.$XML(xmlDoc).find("NetworkInterface", true);
                                for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                                    arrIP.push({
                                        ipv4: NS.$XML(xmlDoc).find("ipAddress").eq(0).text(),
                                        ipv6: NS.$XML(xmlDoc).find("ipv6Address").eq(0).text()
                                    });
                                    break
                                }
                                resolve(arrIP)
                            },
                            error: function () {
                                resolve(arrIP)
                            }
                        })
                    }
                })
            });
            return oPromise
        };
        var _getPPPoEEnable = function (oDeviceInfo) {
            let oPromise = new Promise(function (resolve) {
                var bEnabled = false;
                oDeviceInfo.oProtocolInc.getPPPoEStatus(oDeviceInfo, {
                    success: function (xmlDoc) {
                        if (NS.$XML(xmlDoc).find("ipAddress", true).length > 0) {
                            bEnabled = true
                        } else if (NS.$XML(xmlDoc).find("ipv6Address", true).length > 0) {
                            bEnabled = true
                        } else {
                            bEnabled = false
                        }
                        resolve(bEnabled)
                    }, error: function () {
                        bEnabled = false;
                        resolve(bEnabled)
                    }
                })
            });
            return oPromise
        };
        var _generateTransCodeXml = function (oTransCodeParam) {
            var oDefaultParam = {TransFrameRate: "", TransResolution: "", TransBitrate: ""};
            m_utilsInc.extend(oDefaultParam, oTransCodeParam);
            if (oDefaultParam.TransFrameRate == "" || oDefaultParam.TransResolution == "" || oDefaultParam.TransBitrate == "") {
                return ""
            }
            var ArraySet = [];
            ArraySet.push("<?xml version='1.0' encoding='UTF-8'?>");
            ArraySet.push("<CompressionInfo>");
            ArraySet.push("<TransFrameRate>" + oDefaultParam.TransFrameRate + "</TransFrameRate>");
            ArraySet.push("<TransResolution>" + oDefaultParam.TransResolution + "</TransResolution>");
            ArraySet.push("<TransBitrate>" + oDefaultParam.TransBitrate + "</TransBitrate>");
            ArraySet.push("</CompressionInfo>");
            return ArraySet.join("")
        };
        var _setDeviceInfo = function (cgiInstance, oDeviceInfo, szIP, iProtocol, iPort, szUserName, szPassword) {
            oDeviceInfo.szIP = szIP;
            if (iProtocol == 2) {
                oDeviceInfo.szHttpProtocol = "https://";
                oDeviceInfo.iHttpsPort = iPort
            } else {
                oDeviceInfo.szHttpProtocol = "http://";
                oDeviceInfo.iHttpPort = iPort
            }
            oDeviceInfo.iCGIPort = iPort;
            oDeviceInfo.szDeviceIdentify = szIP + "_" + iPort;
            oDeviceInfo.iDeviceProtocol = PROTOCOL_DEVICE_ISAPI;
            oDeviceInfo.oProtocolInc = cgiInstance;
            oDeviceInfo.szAuth = m_utilsInc.Base64.encode(":" + szUserName + ":" + szPassword)
        };
        var _doLogin = function (cgiInstance, oDeviceInfo, szIP, iProtocol, iPort, szUserName, szPassword, options) {
            var newOptions = {success: null, error: null};
            m_utilsInc.extend(newOptions, options);
            m_utilsInc.extend(newOptions, {
                success: function (xmlDoc) {
                    _setDeviceInfo(cgiInstance, oDeviceInfo, szIP, iProtocol, iPort, szUserName, szPassword);
                    m_deviceSet.push(oDeviceInfo);
                    _initDeviceInfo(oDeviceInfo).then(() => {
                        if (options.success) {
                            options.success(xmlDoc)
                        }
                    })
                }, error: function (oError) {
                    if (options.error) {
                        options.error(oError)
                    }
                }
            });
            return cgiInstance.digestLogin(szIP, iProtocol, iPort, szUserName, szPassword, newOptions)
        };
        this.I_SupportNoPlugin = function () {
            return false
        };
        this.I_Resize = function (iWidth, iHeight) {
            return m_pluginOBJECT.JS_Resize(iWidth, iHeight)
        };
        this.I_InitPlugin = function (options) {
            m_utilsInc.extend(m_options, options);
            var szDirName = m_utilsInc.getDirName();
            if (szDirName) {
                if ("object" === typeof exports && typeof module !== "undefined") {
                } else if ("function" === typeof define && define.amd) {
                    require([szDirName + "/jsVideoPlugin-1.0.0.min.js"], function (o) {
                        window.JSVideoPlugin = o.JSVideoPlugin;
                        if (options.cbInitPluginComplete) {
                            options.cbInitPluginComplete()
                        }
                    })
                } else {
                    m_utilsInc.loadScript(szDirName + "/jsVideoPlugin-1.0.0.min.js", function () {
                        if (options.cbInitPluginComplete) {
                            options.cbInitPluginComplete()
                        }
                    })
                }
            }
            window.addEventListener("resize", function () {
                if (m_pluginOBJECT !== null) {
                    var oElem = $("#" + m_options.szContainerID);
                    m_pluginOBJECT.JS_Resize(oElem.width(), oElem.height())
                }
            });
            window.addEventListener("unload", function () {
            })
        };
        this.I_InsertOBJECTPlugin = function (szContainerID) {
            return _initPlugin(szContainerID)
        };
        this.I_WriteOBJECT_XHTML = function () {
            return 0
        };
        this.I_OpenFileDlg = async function (iType) {
            let oPromise = new Promise(function (resolve, reject) {
                m_pluginOBJECT.JS_OpenFileBrowser(iType, "").then(szFilePath => {
                    resolve(szFilePath)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetLocalCfg = function () {
            let oPromise = new Promise(function (resolve, reject) {
                m_pluginOBJECT.JS_GetLocalConfig().then(oLocalCofing => {
                    resolve(oLocalCofing)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_SetLocalCfg = function (oLocalCofing) {
            let oPromise = new Promise(function (resolve, reject) {
                m_pluginOBJECT.JS_SetLocalConfig(oLocalCofing).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_Login = function (szIP, iProtocol, iPort, szUserName, szPassword, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var szDeviceIdentify = szIP + "_" + iPort;
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    if (options.error) {
                        options.error({
                            errorCode: ERROR_CODE_LOGIN_REPEATLOGIN,
                            errorMsg: "The device is already login."
                        })
                    }
                    reject({errorCode: ERROR_CODE_LOGIN_REPEATLOGIN, errorMsg: "The device is already login."});
                    return
                }
                var cgiInstance = m_ISAPIProtocol;
                var oDeviceInfo = new deviceInfoClass;
                _doLogin(cgiInstance, oDeviceInfo, szIP, iProtocol, iPort, szUserName, szPassword, options).then(() => {
                    resolve()
                }, oError => {
                    reject(oError)
                })
            });
            return oPromise
        };
        this.I_Logout = function (szDeviceIdentify) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    m_deviceSet.splice(iIndex, 1);
                    resolve()
                }
            });
            return oPromise
        };
        this.I_GetAudioInfo = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.getAudioInfo(oDeviceInfo, options).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_GetDeviceInfo = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.getDeviceInfo(oDeviceInfo, options).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_GetAnalogChannelInfo = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.getAnalogChannelInfo(oDeviceInfo, options).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_GetDigitalChannelInfo = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.getDigitalChannelInfo(oDeviceInfo, options).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_GetZeroChannelInfo = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.getZeroChannelInfo(oDeviceInfo, options).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_StartRealPlay = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                var newOptions = {iWndIndex: m_iSelWnd, iStreamType: 1, iChannelID: 1, bZeroChannel: false};
                if(options.m_iSelWnd!=null){
                    newOptions.iWndIndex = options.m_iSelWnd;
                }
                m_utilsInc.extend(newOptions, options);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    var iWndIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                    if (-1 == iWndIndex) {
                        oDeviceInfo.oProtocolInc.startRealPlay(oDeviceInfo, newOptions).then(function () {
                            if (options.success) {
                                options.success()
                            }
                            resolve()
                        }, function () {
                            if (options.error) {
                                options.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        reject({errorCode: ERROR_CODE_PLAY_NOREPEATPLAY, errorMsg: "The window is already playing."})
                    }
                } else {
                    if (options.error) {
                        options.error(_oNoLoginError)
                    }
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_StartPlay = function (szDeviceIdentify, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                var newOptions = {iWndIndex: m_iSelWnd};
                m_utilsInc.extend(newOptions, options);
                var oDeviceInfo = m_deviceSet[iIndex];
                iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (-1 == iIndex) {
                    oDeviceInfo.oProtocolInc.startPlay(oDeviceInfo, newOptions).then(function () {
                        if (options.success) {
                            options.success()
                        }
                        resolve()
                    }, function () {
                        if (options.error) {
                            options.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    })
                } else {
                    reject({errorCode: ERROR_CODE_PLAY_NOREPEATPLAY, errorMsg: "The window is already playing."})
                }
            });
            return oPromise
        };
        this.I_SetSecretKey = function (szSecretKey) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_SetSecretKey(0, szSecretKey, 1).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetEncryptString = function (szSecretKey) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_GetEncryptString(3, szSecretKey).then(szEncode => {
                    resolve(szEncode)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_Stop = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.bRecord) {
                        m_pluginOBJECT.JS_StopSave(wndInfo.iIndex)
                    }
                    if (wndInfo.bSound) {
                        m_pluginOBJECT.JS_CloseSound()
                    }
                    if (wndInfo.bEZoom) {
                        m_pluginOBJECT.JS_DisableZoom(wndInfo.iIndex)
                    }
                    m_pluginOBJECT.JS_Stop(newOptions.iWndIndex).then(() => {
                        m_wndSet.splice(iIndex, 1);
                        if (newOptions.success) {
                            newOptions.success()
                        }
                        resolve()
                    }, () => {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    })
                } else {
                    resolve()
                }
            });
            return oPromise
        };
        this.I_StopAllPlay = function () {
            let oPromise = new Promise(async function (resolve, reject) {
                m_pluginOBJECT.JS_StopRealPlayAll().then(() => {
                    m_wndSet.length = 0;
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_OpenSound = function (iWndIndex) {
            iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (!wndInfo.bSound) {
                        m_pluginOBJECT.JS_OpenSound(iWndIndex).then(() => {
                            wndInfo.bSound = true;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_CloseSound = function (iWndIndex) {
            iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.bSound) {
                        m_pluginOBJECT.JS_CloseSound().then(() => {
                            wndInfo.bSound = false;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_SetVolume = function (iVolume, iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                var iRet = -1;
                iVolume = parseInt(iVolume, 10);
                if (isNaN(iVolume)) {
                    reject(_oParamsError);
                    return
                }
                if (iVolume < 0 || iVolume > 100) {
                    reject(_oParamsError);
                    return
                }
                iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    m_pluginOBJECT.JS_SetVolume(iWndIndex, iVolume).then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_CapturePic = function (szPicName, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {iWndIndex: m_iSelWnd, bDateDir: true};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    if (".jpg" === szPicName.slice(-4).toLowerCase()) {
                        szPicName = szPicName.slice(0, -4)
                    } else if (".jpeg" === szPicName.slice(-5).toLowerCase()) {
                        szPicName = szPicName.slice(0, -5)
                    }
                    m_pluginOBJECT.JS_CapturePicture(newOptions.iWndIndex, szPicName, newOptions.bDateDir).then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_CapturePicData = function (options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {iWndIndex: m_iSelWnd, bDateDir: true};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    m_pluginOBJECT.JS_GetCaptureData(newOptions.iWndIndex).then(function (data) {
                        resolve(data)
                    }, function (data) {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StartRecord = function (szFileName, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {iWndIndex: m_iSelWnd, bDateDir: true};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (!wndInfo.bRecord) {
                        m_pluginOBJECT.JS_StartSave(newOptions.iWndIndex, szFileName).then(function () {
                            wndInfo.bRecord = true;
                            if (newOptions.success) {
                                newOptions.success()
                            }
                            resolve()
                        }, function () {
                            if (newOptions.error) {
                                newOptions.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    }
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StopRecord = function (options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.bRecord) {
                        m_pluginOBJECT.JS_StopSave(newOptions.iWndIndex).then(function () {
                            wndInfo.bRecord = false;
                            if (newOptions.success) {
                                newOptions.success()
                            }
                            resolve()
                        }, function () {
                            if (newOptions.error) {
                                newOptions.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    }
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StartVoiceTalk = function (szDeviceIdentify, iAudioChannel) {
            let oPromise = new Promise((resolve, reject) => {
                if (isNaN(parseInt(iAudioChannel, 10))) {
                    reject(_oParamsError);
                    return
                }
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    if (!oDeviceInfo.bVoiceTalk) {
                        oDeviceInfo.oProtocolInc.startVoiceTalk(oDeviceInfo, iAudioChannel).then(() => {
                            m_deviceSet[iIndex].bVoiceTalk = true;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StopVoiceTalk = function () {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_StopTalk().then(() => {
                    for (var i = 0, iLen = m_deviceSet.length; i < iLen; i++) {
                        if (m_deviceSet[i].bVoiceTalk) {
                            m_deviceSet[i].bVoiceTalk = false;
                            break
                        }
                    }
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_StartAudioPlay = function (szDeviceIdentify, options) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    options.szAuth = oDeviceInfo.szAuth;
                    if (!oDeviceInfo.bVoiceTalk) {
                        oDeviceInfo.oProtocolInc.audioPlay(options).then(() => {
                            m_deviceSet[iIndex].bVoiceTalk = true;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StopAudioPlay = function () {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_StopAudioPlay().then(() => {
                    for (var i = 0, iLen = m_deviceSet.length; i < iLen; i++) {
                        if (m_deviceSet[i].bVoiceTalk) {
                            m_deviceSet[i].bVoiceTalk = false;
                            break
                        }
                    }
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_PTZControl = function (iPTZIndex, bStop, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {iWndIndex: m_iSelWnd, iPTZIndex: iPTZIndex, iPTZSpeed: 4};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    iIndex = this.findDeviceIndexByIP(wndInfo.szIP);
                    if (iIndex != -1) {
                        var oDeviceInfo = m_deviceSet[iIndex];
                        if (9 == iPTZIndex) {
                            oDeviceInfo.oProtocolInc.ptzAutoControl(oDeviceInfo, bStop, wndInfo, newOptions).then(() => {
                                resolve()
                            }, oError => {
                                reject(oError)
                            })
                        } else {
                            oDeviceInfo.oProtocolInc.ptzControl(oDeviceInfo, bStop, wndInfo, newOptions).then(() => {
                                resolve()
                            }, oError => {
                                reject(oError)
                            })
                        }
                    }
                }
            });
            return oPromise
        };
        this.I_EnableEZoom = function (iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (!wndInfo.bEZoom) {
                        m_pluginOBJECT.JS_EnableZoom(iWndIndex).then(() => {
                            wndInfo.bEZoom = true;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_DisableEZoom = function (iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.bEZoom) {
                        m_pluginOBJECT.JS_DisableZoom(iWndIndex).then(() => {
                            wndInfo.bEZoom = false;
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    } else {
                        resolve()
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_Enable3DZoom = function (iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (!wndInfo.b3DZoom) {
                        m_pluginOBJECT.JS_SetDrawCallback(iWndIndex, true, "Rect", false, function (oRect) {
                            _onZoomInfoCallback(oRect.points)
                        });
                        wndInfo.b3DZoom = true;
                        resolve()
                    } else {
                        resolve()
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_Disable3DZoom = function (iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                iWndIndex = m_utilsInc.isUndefined(iWndIndex) ? m_iSelWnd : iWndIndex;
                var iIndex = this.findWndIndexByIndex(iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.b3DZoom) {
                        m_pluginOBJECT.JS_SetDrawCallback(iWndIndex, false, "Rect", false, function () {
                        });
                        wndInfo.b3DZoom = false;
                        resolve()
                    } else {
                        resolve()
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_FullScreen = function (bFull) {
            let oPromise = new Promise(function (resolve, reject) {
                m_pluginOBJECT.JS_FullScreenDisplay(bFull).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_SetPreset = function (iPresetID, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd, iPresetID: iPresetID};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    iIndex = this.findDeviceIndexByIP(wndInfo.szIP);
                    if (iIndex != -1) {
                        var oDeviceInfo = m_deviceSet[iIndex];
                        oDeviceInfo.oProtocolInc.setPreset(oDeviceInfo, wndInfo, newOptions).then(() => {
                            resolve()
                        }, oError => {
                            reject(oError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_GoPreset = function (iPresetID, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd, iPresetID: iPresetID};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    iIndex = this.findDeviceIndexByIP(wndInfo.szIP);
                    if (iIndex != -1) {
                        var oDeviceInfo = m_deviceSet[iIndex];
                        oDeviceInfo.oProtocolInc.goPreset(oDeviceInfo, wndInfo, newOptions).then(() => {
                            resolve()
                        }, oError => {
                            reject(oError)
                        })
                    } else {
                        reject(_oUnKnownError)
                    }
                } else {
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_RecordSearch = function (szDeviceIdentify, iChannelID, szStartTime, szEndTime, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    if (oDeviceInfo.szDeviceType === DEVICE_TYPE_IPCAMERA || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPDOME || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPZOOM) {
                        szStartTime = m_utilsInc.convertToUTCTime(szStartTime);
                        szEndTime = m_utilsInc.convertToUTCTime(szEndTime)
                    }
                    var newOptions = {
                        iChannelID: iChannelID,
                        szStartTime: szStartTime,
                        szEndTime: szEndTime,
                        iSearchPos: 0,
                        iStreamType: 1
                    };
                    m_utilsInc.extend(newOptions, options);
                    newOptions.success = null;
                    oDeviceInfo.oProtocolInc.recordSearch(oDeviceInfo, newOptions).then(oData => {
                        if (oDeviceInfo.szDeviceType === DEVICE_TYPE_IPCAMERA || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPDOME || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPZOOM) {
                            var szRecordStartTime = "";
                            var szRecordEndTime = "";
                            for (var i = 0, nLen = $(oData).find("searchMatchItem").length; i < nLen; i++) {
                                szRecordStartTime = $(oData).find("startTime").eq(i).text();
                                szRecordEndTime = $(oData).find("endTime").eq(i).text();
                                szRecordStartTime = m_utilsInc.convertToLocalTime(szRecordStartTime, oDeviceInfo.iDeviceMinusLocalTime);
                                szRecordEndTime = m_utilsInc.convertToLocalTime(szRecordEndTime, oDeviceInfo.iDeviceMinusLocalTime);
                                $(oData).find("startTime").eq(i).text(szRecordStartTime);
                                $(oData).find("endTime").eq(i).text(szRecordEndTime)
                            }
                        }
                        if (options.success) {
                            options.success(oData)
                        }
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_StartPlayback = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify), cgi = "", urlProtocol = "", iChannelID = 1,
                    iStream = 0;
                var szCurTime = m_utilsInc.dateFormat(new Date, "yyyy-MM-dd");
                var newOptions = {
                    iWndIndex: m_iSelWnd,
                    iStreamType: 1,
                    iChannelID: 1,
                    szStartTime: szCurTime + " 00:00:00",
                    szEndTime: szCurTime + " 23:59:59"
                };
                m_utilsInc.extend(newOptions, options);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    cgi = oDeviceInfo.oProtocolInc.CGI.startPlayback;
                    urlProtocol = "rtsp://";
                    iStream = newOptions.iStreamType;
                    iChannelID = newOptions.iChannelID * 100 + iStream;
                    m_utilsInc.extend(newOptions, {urlProtocol: urlProtocol, cgi: cgi, iChannelID: iChannelID});
                    iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                    if (-1 == iIndex) {
                        if (oDeviceInfo.szDeviceType === DEVICE_TYPE_IPCAMERA || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPDOME || oDeviceInfo.szDeviceType === DEVICE_TYPE_IPZOOM) {
                            newOptions.szStartTime = m_utilsInc.convertToUTCTime(newOptions.szStartTime);
                            newOptions.szEndTime = m_utilsInc.convertToUTCTime(newOptions.szEndTime)
                        }
                        newOptions.szStartTime = newOptions.szStartTime.replace(/[-:]/g, "").replace(" ", "T") + "Z";
                        newOptions.szEndTime = newOptions.szEndTime.replace(/[-:]/g, "").replace(" ", "T") + "Z";
                        oDeviceInfo.oProtocolInc.startPlayback(oDeviceInfo, newOptions).then(function () {
                            if (options.success) {
                                options.success()
                            }
                            resolve()
                        }, function () {
                            if (options.error) {
                                options.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    }
                } else {
                    if (options.error) {
                        options.error(_oNoLoginError)
                    }
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_ReversePlayback = function (szDeviceIdentify, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify), iRet = -1, cgi = "", urlProtocol = "",
                    iPort = -1, iChannelID = -1, iStream = 0;
                var szCurTime = m_utilsInc.dateFormat(new Date, "yyyy-MM-dd");
                var newOptions = {
                    iWndIndex: m_iSelWnd,
                    iStreamType: 1,
                    iChannelID: 1,
                    szStartTime: szCurTime + " 00:00:00",
                    szEndTime: szCurTime + " 23:59:59"
                };
                m_utilsInc.extend(newOptions, options);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    var iProtocolType = parseInt(m_oLocalCfg.protocolType, 10);
                    cgi = oDeviceInfo.oProtocolInc.CGI.startPlayback;
                    urlProtocol = "rtsp://";
                    iStream = newOptions.iStreamType;
                    iChannelID = newOptions.iChannelID * 100 + iStream;
                    m_utilsInc.extend(newOptions, {urlProtocol: urlProtocol, cgi: cgi, iChannelID: iChannelID});
                    iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                    if (-1 == iIndex) {
                        newOptions.szStartTime = newOptions.szStartTime.replace(/[-:]/g, "").replace(" ", "T") + "Z";
                        newOptions.szEndTime = newOptions.szEndTime.replace(/[-:]/g, "").replace(" ", "T") + "Z";
                        oDeviceInfo.oProtocolInc.reversePlayback(oDeviceInfo, newOptions).then(function () {
                            if (options.success) {
                                options.success()
                            }
                            resolve()
                        }, function () {
                            if (options.error) {
                                options.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    }
                }
            });
            return oPromise
        };
        this.I_Frame = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex], iPlayStatus = wndInfo.iPlayStatus;
                    if (iPlayStatus == PLAY_STATUS_PLAYBACK || iPlayStatus == PLAY_STATUS_FRAME) {
                        m_pluginOBJECT.JS_FrameForward(newOptions.iWndIndex).then(function () {
                            wndInfo.iPlayStatus = PLAY_STATUS_FRAME;
                            if (newOptions.success) {
                                newOptions.success()
                            }
                            resolve()
                        }, function () {
                            if (newOptions.error) {
                                newOptions.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    }
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_Pause = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex], iPlayStatus = wndInfo.iPlayStatus, iNextStatus = -1;
                    if (iPlayStatus == PLAY_STATUS_PLAYBACK) {
                        iNextStatus = PLAY_STATUS_PAUSE
                    } else if (iPlayStatus == PLAY_STATUS_REVERSE_PLAYBACK) {
                        iNextStatus = PLAY_STATUS_REVERSE_PAUSE
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError);
                        return
                    }
                    m_pluginOBJECT.JS_Pause(newOptions.iWndIndex).then(function () {
                        wndInfo.iPlayStatus = iNextStatus;
                        if (newOptions.success) {
                            newOptions.success()
                        }
                        resolve()
                    }, function () {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    })
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_Resume = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex], iPlayStatus = wndInfo.iPlayStatus, iNextStatus = -1;
                    if (iPlayStatus == PLAY_STATUS_PAUSE || iPlayStatus == PLAY_STATUS_FRAME) {
                        iNextStatus = PLAY_STATUS_PLAYBACK
                    } else if (iPlayStatus == PLAY_STATUS_REVERSE_PAUSE) {
                        iNextStatus = PLAY_STATUS_REVERSE_PLAYBACK
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError);
                        return
                    }
                    m_pluginOBJECT.JS_Resume(newOptions.iWndIndex).then(function () {
                        wndInfo.iPlayStatus = iNextStatus;
                        if (newOptions.success) {
                            newOptions.success()
                        }
                        resolve()
                    }, function () {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    })
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_PlaySlow = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.iPlayStatus == PLAY_STATUS_PLAYBACK) {
                        m_pluginOBJECT.JS_Slow(newOptions.iWndIndex).then(function () {
                            if (newOptions.success) {
                                newOptions.success()
                            }
                            resolve()
                        }, function () {
                            if (newOptions.error) {
                                newOptions.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    }
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_PlayFast = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    var wndInfo = m_wndSet[iIndex];
                    if (wndInfo.iPlayStatus == PLAY_STATUS_PLAYBACK) {
                        m_pluginOBJECT.JS_Fast(newOptions.iWndIndex).then(function () {
                            if (newOptions.success) {
                                newOptions.success()
                            }
                            resolve()
                        }, function () {
                            if (newOptions.error) {
                                newOptions.error(_oUnKnownError)
                            }
                            reject(_oUnKnownError)
                        })
                    } else {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    }
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_GetOSDTime = function (options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var newOptions = {iWndIndex: m_iSelWnd};
                if (m_utilsInc.isObject(options)) {
                    m_utilsInc.extend(newOptions, options)
                } else {
                    if (!m_utilsInc.isUndefined(options)) {
                        newOptions.iWndIndex = options
                    }
                }
                var iIndex = this.findWndIndexByIndex(newOptions.iWndIndex);
                if (iIndex != -1) {
                    m_pluginOBJECT.JS_GetOSDTime(newOptions.iWndIndex).then(function (iTime) {
                        if (newOptions.success) {
                            var szOSDTime = m_utilsInc.dateFormat(new Date(iTime * 1e3), "yyyy-MM-dd hh:mm:ss");
                            newOptions.success(szOSDTime)
                        }
                        resolve(szOSDTime)
                    }, function () {
                        if (newOptions.error) {
                            newOptions.error(_oUnKnownError)
                        }
                        reject(_oUnKnownError)
                    })
                } else {
                    if (newOptions.error) {
                        newOptions.error(_oUnKnownError)
                    }
                    reject(_oUnKnownError)
                }
            });
            return oPromise
        };
        this.I_StartDownloadRecord = function (szDeviceIdentify, szPlaybackURI, szFileName, options) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    var newOptions = {szPlaybackURI: szPlaybackURI, szFileName: szFileName + ".mp4", bDateDir: true};
                    if (!m_utilsInc.isUndefined(options)) {
                        m_utilsInc.extend(newOptions, options)
                    }
                    oDeviceInfo.oProtocolInc.startDownloadRecord(oDeviceInfo, newOptions).then(iDownloadID => {
                        resolve(iDownloadID)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_StartDownloadRecordByTime = function (szDeviceIdentify, szPlaybackURI, szFileName, szStartTime, szEndTime, options) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    szPlaybackURI = szPlaybackURI.split("?")[0] + "?starttime=" + szStartTime.replace(" ", "T") + "Z&endtime=" + szEndTime.replace(" ", "T") + "Z";
                    var newOptions = {szPlaybackURI: szPlaybackURI, szFileName: szFileName + ".mp4", bDateDir: true};
                    if (!m_utilsInc.isUndefined(options)) {
                        m_utilsInc.extend(newOptions, options)
                    }
                    oDeviceInfo.oProtocolInc.startDownloadRecord(oDeviceInfo, newOptions).then(iDownloadID => {
                        resolve(iDownloadID)
                    }, oError => {
                        reject(oError)
                    })
                }
            });
            return oPromise
        };
        this.I_GetDownloadStatus = function (iDownloadID) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_GetDownloadStatus(iDownloadID).then(data => {
                    resolve(data)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetDownloadProgress = function (iDownloadID) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_GetDownloadProgress(iDownloadID).then(data => {
                    resolve(data)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_StopDownloadRecord = function (iDownloadID) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_StopAsyncDownload(iDownloadID).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_ExportDeviceConfig = function (szDeviceIdentify) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.exportDeviceConfig(oDeviceInfo).then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_ImportDeviceConfig = function (szDeviceIdentify, szFileName) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    var newOptions = {szFileName: szFileName};
                    oDeviceInfo.oProtocolInc.importDeviceConfig(oDeviceInfo, newOptions).then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_RestoreDefault = function (szDeviceIdentify, szMode, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {success: null, error: null};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.restore(oDeviceInfo, szMode, newOptions).then(() => {
                        resolve()
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_Restart = function (szDeviceIdentify, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {success: null, error: null};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.restart(oDeviceInfo, newOptions).then(() => {
                        resolve()
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_Reconnect = function (szDeviceIdentify, options) {
            let oPromise = new Promise((resolve, reject) => {
                var newOptions = {success: null, error: null};
                m_utilsInc.extend(newOptions, options);
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.login(oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oDeviceInfo.szAuth, newOptions).then(() => {
                        resolve()
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_StartUpgrade = function (szDeviceIdentify, szFileName) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    oDeviceInfo.oProtocolInc.startUpgrade(oDeviceInfo, szFileName).then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_UpgradeStatus = function (szDeviceIdentify) {
            let oPromise = new Promise((resolve, reject) => {
                this.I_SendHTTPRequest(szDeviceIdentify, m_ISAPIProtocol.CGI.startUpgrade.status, {}).then(data => {
                    var bUpgrading = $(data).find("upgrading").eq(0).text() === "true";
                    resolve(bUpgrading)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_UpgradeProgress = function (szDeviceIdentify) {
            let oPromise = new Promise((resolve, reject) => {
                this.I_SendHTTPRequest(szDeviceIdentify, m_ISAPIProtocol.CGI.startUpgrade.status, {}).then(data => {
                    var iPercent = parseInt($(data).find("percent").eq(0).text(), 10);
                    resolve(iPercent)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_StopUpgrade = function () {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_StopUpgrade().then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_CheckPluginInstall = function () {
            return true
        };
        this.I_CheckPluginVersion = function () {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_CheckUpdate(m_szVersion).then(bFlag => {
                    resolve(bFlag)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_SendHTTPRequest = function (szDeviceIdentify, szURI, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var httpClient = new HttpPluginClient;
                var szURL = "";
                var szAuth = "";
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex >= 0) {
                    if ("%" === szURI.substr(0, 1)) {
                        szURI = szURI.substr(8)
                    }
                    var oDeviceInfo = m_deviceSet[iIndex];
                    szURL = oDeviceInfo.szHttpProtocol + oDeviceInfo.szIP + ":" + oDeviceInfo.iCGIPort + "/" + szURI;
                    szAuth = oDeviceInfo.szAuth
                }
                var newOptions = {type: "GET", url: szURL, auth: szAuth, success: null, error: null};
                m_utilsInc.extend(newOptions, options);
                httpClient.submitRequest(newOptions).then(function (oRes) {
                    if (200 === oRes.httpStatusCode) {
                        let oData;
                        if (0 === oRes.httpResponse.indexOf("<?xml")) {
                            oData = m_utilsInc.loadXML(oRes.httpResponse)
                        } else {
                            oData = JSON.parse(oRes.httpResponse)
                        }
                        options.success && options.success(oData);
                        resolve(oData)
                    } else if (200 !== oRes.httpStatusCode) {
                        let oData = m_utilsInc.loadXML(oRes.httpResponse);
                        if (!oData) {
                            oData = JSON.parse(oRes.httpResponse)
                        }
                        options.error && options.error({errorCode: oRes.httpStatusCode, errorMsg: oData});
                        reject({errorCode: oRes.httpStatusCode, errorMsg: oData})
                    }
                }, function (errorCode) {
                    if (options.error) {
                        options.error({errorCode: errorCode, errorMsg: ""})
                    }
                    reject({errorCode: errorCode, errorMsg: ""})
                })
            });
            return oPromise
        };
        this.I_ChangeWndNum = function (iWndType) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_ArrangeWindow(iWndType).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetLastError = function () {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_GetLastError().then(data => {
                    resolve(data)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetWindowStatus = function (iWndIndex) {
            if (m_utilsInc.isUndefined(iWndIndex)) {
                var wndSet = [];
                m_utilsInc.extend(wndSet, m_wndSet);
                return wndSet
            } else {
                var i = this.findWndIndexByIndex(iWndIndex);
                if (i != -1) {
                    var wndSet = {};
                    m_utilsInc.extend(wndSet, m_wndSet[i]);
                    return wndSet
                } else {
                    return null
                }
            }
        };
        this.I_GetIPInfoByMode = function (iMode, szAddress, iPort, szDeviceInfo) {
            return
        };
        this.I_SetPlayModeType = function (iMode) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_SetPlayMode(iMode).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_SetSnapDrawMode = function (iWndIndex, iMode) {
            let bType = false;
            if (iMode !== -1) {
                bType = true
            }
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_SetDrawStatus(bType, iMode).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_SetSnapPolygonInfo = function (iWndIndex, szInfo) {
            let oPromise = new Promise((resolve, reject) => {
                var aP = [];
                var oData = m_utilsInc.formatPolygonXmlToJson(szInfo);
                if (oData.aAddRect.length > 0) {
                    aP.push(m_pluginOBJECT.JS_SetDrawShapeInfo("Rect", oData.aAddRect[0]))
                }
                if (oData.aAddPolygon.length > 0) {
                    aP.push(m_pluginOBJECT.JS_SetDrawShapeInfo("Polygon", oData.aAddPolygon[0]))
                }
                if (oData.aRect.length > 0) {
                    aP.push(m_pluginOBJECT.JS_SetRectInfo(oData.aRect))
                }
                if (oData.aPolygon.length > 0) {
                    aP.push(m_pluginOBJECT.JS_SetPolygonInfo(oData.aPolygon))
                }
                Promise.all(aP).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetSnapPolygonInfo = function (iWndIndex) {
            let oPromise = new Promise((resolve, reject) => {
                var aP = [];
                aP.push(m_pluginOBJECT.JS_GetPolygonInfo());
                aP.push(m_pluginOBJECT.JS_GetRectInfo());
                Promise.all(aP).then(aData => {
                    var szXmlData = m_utilsInc.formatPolygonJsonToXml(aData);
                    resolve(szXmlData)
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_ClearSnapInfo = function (iWndIndex, aShapes) {
            let oPromise = new Promise((resolve, reject) => {
                if (aShapes) {
                    var aPolygon = [];
                    var aRect = [];
                    aShapes.forEach(item => {
                        if (1 === item.polygonType) {
                            aPolygon.push(item.id)
                        } else {
                            aRect.push(item.id)
                        }
                        var aP = [];
                        if (aPolygon.length) {
                            aP.push(m_pluginOBJECT.JS_ClearShapeByType("Polygon", aPolygon))
                        }
                        if (aRect.length) {
                            aP.push(m_pluginOBJECT.JS_ClearShapeByType("Rect", aRect))
                        }
                        Promise.all(aP).then(() => {
                            resolve()
                        }, () => {
                            reject(_oUnKnownError)
                        })
                    })
                } else {
                    m_pluginOBJECT.JS_ClearShapeByType("AllWindows").then(() => {
                        resolve()
                    }, () => {
                        reject(_oUnKnownError)
                    })
                }
            });
            return oPromise
        };
        this.I_DeviceCapturePic = function (szDeviceIdentify, iChannelID, szPicName, options) {
            return false;
            var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
            var iRet = -1;
            if (iIndex != -1) {
                var oDeviceInfo = m_deviceSet[iIndex];
                var newOptions = {bDateDir: true};
                m_utilsInc.extend(newOptions, options);
                if (!m_utilsInc.isUndefined(newOptions.iResolutionWidth) && !m_utilsInc.isInt(newOptions.iResolutionWidth)) {
                    return iRet
                }
                if (!m_utilsInc.isUndefined(newOptions.iResolutionHeight) && !m_utilsInc.isInt(newOptions.iResolutionHeight)) {
                    return iRet
                }
                iRet = oDeviceInfo.oProtocolInc.deviceCapturePic(oDeviceInfo, iChannelID, szPicName, newOptions)
            }
            return iRet
        };
        this.I_SetPackageType = function (iPackageType) {
            let oPromise = new Promise((resolve, reject) => {
                m_pluginOBJECT.JS_SetPackageType(iPackageType).then(() => {
                    resolve()
                }, () => {
                    reject(_oUnKnownError)
                })
            });
            return oPromise
        };
        this.I_GetDevicePort = function (szDeviceIdentify) {
            let oPromise = new Promise(async (resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                var oPort = null;
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    try {
                        oPort = await _getPort(oDeviceInfo);
                        resolve(oPort)
                    } catch (err) {
                        reject({errorCode: ERROR_CODE_NETWORKERROR, errorMsg: ""})
                    }
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.I_GetTextOverlay = function (szUrl, szDeviceIdentify, options) {
            let oPromise = new Promise((resolve, reject) => {
                var iIndex = this.findDeviceIndexByIP(szDeviceIdentify);
                if (iIndex != -1) {
                    var oDeviceInfo = m_deviceSet[iIndex];
                    var wndInfo = m_wndSet[iIndex];
                    var newOptions = {type: "GET", success: options.success, error: options.error};
                    this.I_SendHTTPRequest(oDeviceInfo.szIP + "_" + oDeviceInfo.iCGIPort, szUrl, newOptions).then(oData => {
                        resolve(oData)
                    }, oError => {
                        reject(oError)
                    })
                } else {
                    reject(_oNoLoginError)
                }
            });
            return oPromise
        };
        this.findDeviceIndexByIP = function (szDeviceIdentify) {
            if (szDeviceIdentify.indexOf("_") > -1) {
                for (var i = 0, iLen = m_deviceSet.length; i < iLen; i++) {
                    if (m_deviceSet[i].szDeviceIdentify == szDeviceIdentify) {
                        return i
                    }
                }
            } else {
                for (var i = 0, iLen = m_deviceSet.length; i < iLen; i++) {
                    if (m_deviceSet[i].szIP == szDeviceIdentify) {
                        return i
                    }
                }
            }
            return -1
        };
        this.findWndIndexByIndex = function (iWndIndex) {
            for (var i = 0, iLen = m_wndSet.length; i < iLen; i++) {
                if (m_wndSet[i].iIndex == iWndIndex) {
                    return i
                }
            }
            return -1
        };

        function deviceInfoClass() {
            this.szIP = "";
            this.szHostName = "";
            this.szAuth = "";
            this.szHttpProtocol = "http://";
            this.iCGIPort = 80;
            this.szDeviceIdentify = "";
            this.iDevicePort = -1;
            this.iHttpPort = -1;
            this.iHttpsPort = -1;
            this.iRtspPort = -1;
            this.iAudioType = 1;
            this.m_iAudioBitRate = -1;
            this.m_iAudioSamplingRate = -1;
            this.iDeviceProtocol = PROTOCOL_DEVICE_ISAPI;
            this.oProtocolInc = null;
            this.iAnalogChannelNum = 0;
            this.szDeviceType = "";
            this.bVoiceTalk = false;
            this.iDeviceMinusLocalTime = 0
        }

        var wndInfoClass = function () {
            this.iIndex = 0;
            this.szIP = "";
            this.iCGIPort = 80;
            this.szDeviceIdentify = "";
            this.iChannelID = "";
            this.iPlayStatus = PLAY_STATUS_STOP;
            this.bSound = false;
            this.bRecord = false;
            this.bPTZAuto = false;
            this.bEZoom = false;
            this.b3DZoom = false
        };
        var HttpPluginClient = function () {
            this.options = {
                type: "GET",
                url: "",
                auth: "",
                timeout: 3e4,
                data: "",
                async: true,
                success: null,
                error: null
            };
            this.m_szHttpHead = "";
            this.m_szHttpContent = "";
            this.m_szHttpData = ""
        };
        HttpPluginClient.prototype.submitRequest = function (options) {
            options.method = this.getHttpMethod(options.type);
            options.content = options.data;
            delete options.type;
            delete options.data;
            return m_pluginOBJECT.JS_SubmitHttpRequest(options)
        };
        HttpPluginClient.prototype.getHttpMethod = function (szMethod) {
            var oMethod = {GET: 1, POST: 2, PUT: 5, DELETE: 6}, iMethod = oMethod[szMethod];
            return iMethod ? iMethod : -1
        };
        var ISAPIProtocol = function () {
        };
        ISAPIProtocol.prototype.CGI = {
            login: "%s%s:%s/ISAPI/Security/userCheck?format=json",
            getAudioInfo: "%s%s:%s/ISAPI/System/TwoWayAudio/channels",
            getDeviceInfo: "%s%s:%s/ISAPI/System/deviceInfo",
            getAnalogChannelInfo: "%s%s:%s/ISAPI/System/Video/inputs/channels",
            getDigitalChannel: "%s%s:%s/ISAPI/ContentMgmt/InputProxy/channels",
            getDigitalChannelInfo: "%s%s:%s/ISAPI/ContentMgmt/InputProxy/channels/status",
            getZeroChannelInfo: "%s%s:%s/ISAPI/ContentMgmt/ZeroVideo/channels",
            getStreamChannels: {
                analog: "%s%s:%s/ISAPI/Streaming/channels",
                digital: "%s%s:%s/ISAPI/ContentMgmt/StreamingProxy/channels"
            },
            startRealPlay: {
                channels: "video://%s:%s/%s",
                zeroChannels: "%s%s:%s/PSIA/Custom/SelfExt/ContentMgmt/ZeroStreaming/channels/%s"
            },
            startVoiceTalk: {
                open: "%s%s:%s/ISAPI/System/TwoWayAudio/channels/%s/open",
                close: "%s%s:%s/ISAPI/System/TwoWayAudio/channels/%s/close",
                audioData: "%s%s:%s/ISAPI/System/TwoWayAudio/channels/%s/audioData"
            },
            ptzControl: {
                analog: "%s%s:%s/ISAPI/PTZCtrl/channels/%s/continuous",
                digital: "%s%s:%s/ISAPI/ContentMgmt/PTZCtrlProxy/channels/%s/continuous"
            },
            ptzAutoControl: {
                analog: "%s%s:%s/ISAPI/PTZCtrl/channels/%s/autoPan",
                digital: "%s%s:%s/ISAPI/ContentMgmt/PTZCtrlProxy/channels/%s/autoPan"
            },
            setPreset: {
                analog: "%s%s:%s/ISAPI/PTZCtrl/channels/%s/presets/%s",
                digital: "%s%s:%s/ISAPI/ContentMgmt/PTZCtrlProxy/channels/%s/presets/%s"
            },
            goPreset: {
                analog: "%s%s:%s/ISAPI/PTZCtrl/channels/%s/presets/%s/goto",
                digital: "%s%s:%s/ISAPI/ContentMgmt/PTZCtrlProxy/channels/%s/presets/%s/goto"
            },
            ptzFocus: {
                analog: "%s%s:%s/ISAPI/System/Video/inputs/channels/%s/focus",
                digital: "%s%s:%s/ISAPI/ContentMgmt/ImageProxy/channels/%s/focus"
            },
            ptzIris: {
                analog: "%s%s:%s/ISAPI/System/Video/inputs/channels/%s/iris",
                digital: "%s%s:%s/ISAPI/ContentMgmt/ImageProxy/channels/%s/iris"
            },
            getNetworkBond: "%s%s:%s/ISAPI/System/Network/Bond",
            getNetworkInterface: "%s%s:%s/ISAPI/System/Network/interfaces",
            getUPnPPortStatus: "%s%s:%s/ISAPI/System/Network/UPnP/ports/status",
            getPPPoEStatus: "%s%s:%s/ISAPI/System/Network/PPPoE/1/status",
            getPortInfo: "%s%s:%s/ISAPI/Security/adminAccesses",
            recordSearch: "%s%s:%s/ISAPI/ContentMgmt/search",
            startPlayback: "video://%s:%s/%s",
            startWsPlayback: "%s%s:%s/%s",
            startShttpPlayback: "%s%s:%s/SDK/playback/%s",
            startShttpReversePlayback: "%s%s:%s/SDK/playback/%s/reversePlay",
            startTransCodePlayback: "%s%s:%s/SDK/playback/%s/transcoding",
            startDownloadRecord: "%s%s:%s/ISAPI/ContentMgmt/download",
            downloaddeviceConfig: "%s%s:%s/ISAPI/System/configurationData",
            uploaddeviceConfig: "%s%s:%s/ISAPI/System/configurationData",
            restart: "%s%s:%s/ISAPI/System/reboot",
            restore: "%s%s:%s/ISAPI/System/factoryReset?mode=%s",
            startUpgrade: {
                upgrade: "%s%s:%s/ISAPI/System/updateFirmware",
                status: "%s%s:%s/ISAPI/System/upgradeStatus"
            },
            set3DZoom: {
                analog: "%s%s:%s/ISAPI/PTZCtrl/channels/%s/position3D",
                digital: "%s%s:%s/ISAPI/ContentMgmt/PTZCtrlProxy/channels/%s/position3D"
            },
            getSecurityVersion: "%s%s:%s/ISAPI/Security/capabilities?username=admin",
            SDKCapabilities: "%s%s:%s/SDK/capabilities",
            deviceCapture: {channels: "%s%s:%s/ISAPI/Streaming/channels/%s/picture"},
            overlayInfo: {
                analog: "%s%s:%s/ISAPI/System/Video/inputs/channels/%s/overlays/",
                digital: "%s%s:%s/ISAPI/ContentMgmt/InputProxy/channels/%s/video/overlays"
            },
            sessionCap: "%s%s:%s/ISAPI/Security/sessionLogin/capabilities?username=%s",
            sessionLogin: "%s%s:%s/ISAPI/Security/sessionLogin",
            sessionHeartbeat: "%s%s:%s/ISAPI/Security/sessionHeartbeat",
            sessionLogout: "%s%s:%s/ISAPI/Security/sessionLogout",
            systemCapabilities: "%s%s:%s/ISAPI/System/capabilities",
            time: "ISAPI/System/time"
        };
        ISAPIProtocol.prototype.login = function (szIP, iPort, szAuth, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.login, options)
        };
        ISAPIProtocol.prototype.getAudioInfo = function (oDeviceInfo, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (oData) {
                        var oNodeList = NS.$XML(oData).find("audioCompressionType", true);
                        if (oNodeList.length > 0) {
                            var szAudioCompressionType = NS.$XML(oNodeList).eq(0).text(), iAudioType = 0;
                            if ("G.711ulaw" == szAudioCompressionType) {
                                iAudioType = 1
                            } else if ("G.711alaw" == szAudioCompressionType) {
                                iAudioType = 2
                            } else if ("G.726" == szAudioCompressionType) {
                                iAudioType = 3
                            } else if ("MP2L2" == szAudioCompressionType || "MPEL2" == szAudioCompressionType) {
                                iAudioType = 4
                            } else if ("G.722.1" == szAudioCompressionType) {
                                iAudioType = 0
                            } else if ("AAC" == szAudioCompressionType) {
                                iAudioType = 5
                            } else if ("PCM" == szAudioCompressionType) {
                                iAudioType = 6
                            } else if ("MP3" == szAudioCompressionType) {
                                iAudioType = 7
                            }
                            oDeviceInfo.iAudioType = iAudioType
                        }
                        if (NS.$XML(oData).find("audioBitRate").eq(0).text() !== "") {
                            oDeviceInfo.m_iAudioBitRate = parseInt(NS.$XML(oData).find("audioBitRate").eq(0).text(), 10) * 1e3
                        } else {
                            oDeviceInfo.m_iAudioBitRate = 0
                        }
                        if (NS.$XML(oData).find("audioSamplingRate").eq(0).text() !== "") {
                            oDeviceInfo.m_iAudioSamplingRate = parseInt(NS.$XML(oData).find("audioSamplingRate").eq(0).text(), 10) * 1e3
                        } else {
                            oDeviceInfo.m_iAudioSamplingRate = 0
                        }
                        if (NS.$XML(oData).find("channelNum").eq(0).text() !== "") {
                            oDeviceInfo.m_iSoundChan = parseInt(NS.$XML(oData).find("channelNum").eq(0).text(), 10)
                        } else {
                            oDeviceInfo.m_iSoundChan = 1
                        }
                        if (NS.$XML(oData).find("deviceCastChannelNum").eq(0).text() !== "") {
                            oDeviceInfo.m_iDeviceAudioSoundChan = parseInt(NS.$XML(oData).find("deviceCastChannelNum").eq(0).text(), 10)
                        } else {
                            oDeviceInfo.m_iDeviceAudioSoundChan = 1
                        }
                        if (options.success) {
                            options.success(oData)
                        }
                        resolve(oData)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.getAudioInfo, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getDeviceInfo = function (oDeviceInfo, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (xmlDoc) {
                        var oData;
                        oDeviceInfo.szDeviceType = NS.$XML(xmlDoc).find("deviceType").eq(0).text();
                        var arrXml = [];
                        arrXml.push("<DeviceInfo>");
                        arrXml.push("<deviceName>" + m_utilsInc.escape(NS.$XML(xmlDoc).find("deviceName").eq(0).text()) + "</deviceName>");
                        arrXml.push("<deviceID>" + NS.$XML(xmlDoc).find("deviceID").eq(0).text() + "</deviceID>");
                        arrXml.push("<deviceType>" + NS.$XML(xmlDoc).find("deviceType").eq(0).text() + "</deviceType>");
                        arrXml.push("<model>" + NS.$XML(xmlDoc).find("model").eq(0).text() + "</model>");
                        arrXml.push("<serialNumber>" + NS.$XML(xmlDoc).find("serialNumber").eq(0).text() + "</serialNumber>");
                        arrXml.push("<macAddress>" + NS.$XML(xmlDoc).find("macAddress").eq(0).text() + "</macAddress>");
                        arrXml.push("<firmwareVersion>" + NS.$XML(xmlDoc).find("firmwareVersion").eq(0).text() + "</firmwareVersion>");
                        arrXml.push("<firmwareReleasedDate>" + NS.$XML(xmlDoc).find("firmwareReleasedDate").eq(0).text() + "</firmwareReleasedDate>");
                        arrXml.push("<encoderVersion>" + NS.$XML(xmlDoc).find("encoderVersion").eq(0).text() + "</encoderVersion>");
                        arrXml.push("<encoderReleasedDate>" + NS.$XML(xmlDoc).find("encoderReleasedDate").eq(0).text() + "</encoderReleasedDate>");
                        arrXml.push("</DeviceInfo>");
                        oData = m_utilsInc.loadXML(arrXml.join(""));
                        if (options.success) {
                            options.success(oData)
                        }
                        resolve(oData)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.getDeviceInfo, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getDeviceMinusLocalTime = function (oDeviceInfo) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {
                    success: xmlDoc => {
                        var szDeviceTime = $(xmlDoc).find("localTime").eq(0).text().substring(0, 19);
                        var arDTms = szDeviceTime.match(/(\d+)-(\d+)-(\d+)(\D+)(\d+):(\d+):(\d+)/);
                        if (arDTms.length !== 8) {
                            return
                        }
                        var dtDeviceDate = new Date(arDTms[1], arDTms[2] - 1, arDTms[3], arDTms[5], arDTms[6], arDTms[7]);
                        var szTimeZone = $(xmlDoc).find("timeZone").eq(0).text();
                        var iDSTTime = 0;
                        var iDSTPos = szTimeZone.indexOf("DST");
                        if (iDSTPos != -1) {
                            var dtDSTStart = new Date(dtDeviceDate.getTime());
                            dtDSTStart.setMinutes(0);
                            dtDSTStart.setSeconds(0);
                            var dtDSTStop = new Date(dtDeviceDate.getTime());
                            dtDSTStop.setMinutes(0);
                            dtDSTStop.setSeconds(0);
                            var szDSTStartTime = szTimeZone.split(",")[1];
                            var szDSTStopTime = szTimeZone.split(",")[2];
                            var iDSTStartMonth = parseInt(szDSTStartTime.split(".")[0].replace("M", ""), 10);
                            dtDSTStart.setMonth(iDSTStartMonth - 1);
                            var iDSTStartWeek = parseInt(szDSTStartTime.split(".")[1], 10);
                            var iDSTStartDay = parseInt(szDSTStartTime.split(".")[2].split("/")[0]);
                            var iDSTStartTime = parseInt(szDSTStartTime.split(".")[2].split("/")[1].split(":")[0], 10);
                            dtDSTStart.setHours(iDSTStartTime);
                            var iTime = 0;
                            var iDate = 0;
                            for (var i = 1; i <= 31; i++) {
                                dtDSTStart.setDate(i);
                                if (dtDSTStart.getMonth() !== iDSTStartMonth - 1) {
                                    break
                                }
                                if (dtDSTStart.getDay() == iDSTStartDay) {
                                    iTime++;
                                    iDate = i;
                                    if (iTime == iDSTStartWeek) {
                                        break
                                    }
                                }
                            }
                            dtDSTStart.setDate(iDate);
                            dtDSTStart.setMonth(iDSTStartMonth - 1);
                            var iDSTStopMonth = parseInt(szDSTStopTime.split(".")[0].replace("M", ""), 10);
                            dtDSTStop.setMonth(iDSTStopMonth - 1);
                            var iDSTStopWeek = parseInt(szDSTStopTime.split(".")[1], 10);
                            var iDSTStopDay = parseInt(szDSTStopTime.split(".")[2].split("/")[0]);
                            var iDSTStopTime = parseInt(szDSTStopTime.split(".")[2].split("/")[1].split(":")[0], 10);
                            dtDSTStop.setHours(iDSTStopTime);
                            iTime = 0;
                            iDate = 0;
                            for (var i = 1; i <= 31; i++) {
                                dtDSTStop.setDate(i);
                                if (dtDSTStop.getMonth() !== iDSTStopMonth - 1) {
                                    break
                                }
                                if (dtDSTStop.getDay() == iDSTStopDay) {
                                    iTime++;
                                    iDate = i;
                                    if (iTime == iDSTStopWeek) {
                                        break
                                    }
                                }
                            }
                            dtDSTStop.setDate(iDate);
                            dtDSTStop.setMonth(iDSTStopMonth - 1);
                            if (dtDSTStart.getTime() < dtDSTStop.getTime()) {
                                if (dtDeviceDate.getTime() >= dtDSTStart.getTime() && dtDeviceDate.getTime() <= dtDSTStop.getTime()) {
                                    var szDSTTime = szTimeZone.substring(iDSTPos + 3, iDSTPos + 11);
                                    iDSTTime = parseInt(szDSTTime.split(":")[0], 10) * 60 + parseInt(szDSTTime.split(":")[1], 10)
                                }
                            } else {
                                if (dtDeviceDate.getTime() >= dtDSTStart.getTime() || dtDeviceDate.getTime() <= dtDSTStop.getTime()) {
                                    var szDSTTime = szTimeZone.substring(iDSTPos + 3, iDSTPos + 11);
                                    iDSTTime = parseInt(szDSTTime.split(":")[0], 10) * 60 + parseInt(szDSTTime.split(":")[1], 10)
                                }
                            }
                        }
                        var arDTZms = szTimeZone.match(/\D+([+-])(\d+):(\d+):(\d+)/);
                        if (arDTZms.length == 5) {
                            var dtNow = new Date;
                            var iLocalOffsetMin = dtNow.getTimezoneOffset();
                            var iDeviceOffsetMin = parseInt(arDTZms[2]) * 60 + parseInt(arDTZms[3]);
                            iDeviceOffsetMin = arDTZms[1] === "+" ? iDeviceOffsetMin : -iDeviceOffsetMin;
                            iDeviceMinusLocalTime = (iLocalOffsetMin - iDeviceOffsetMin + iDSTTime) * 60 * 1e3
                        }
                        oDeviceInfo.iDeviceMinusLocalTime = iDeviceMinusLocalTime;
                        resolve(iDeviceMinusLocalTime)
                    }, error: () => {
                        reject()
                    }
                };
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.time, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getAnalogChannelInfo = function (oDeviceInfo, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (xmlData) {
                        var arrXml = [];
                        arrXml.push("<VideoInputChannelList>");
                        var nodeList = NS.$XML(xmlData).find("VideoInputChannel", true);
                        oDeviceInfo.iAnalogChannelNum = nodeList.length;
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            var node = nodeList[i];
                            arrXml.push("<VideoInputChannel>");
                            arrXml.push("<id>" + NS.$XML(node).find("id").eq(0).text() + "</id>");
                            arrXml.push("<inputPort>" + NS.$XML(node).find("inputPort").eq(0).text() + "</inputPort>");
                            arrXml.push("<name>" + m_utilsInc.escape(NS.$XML(node).find("name").eq(0).text()) + "</name>");
                            arrXml.push("<videoFormat>" + NS.$XML(node).find("videoFormat").eq(0).text() + "</videoFormat>");
                            arrXml.push("</VideoInputChannel>")
                        }
                        arrXml.push("</VideoInputChannelList>");
                        var xmlDoc = m_utilsInc.loadXML(arrXml.join(""));
                        if (options.success) {
                            options.success(xmlDoc)
                        }
                        resolve(xmlDoc)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.getAnalogChannelInfo, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getDigitalChannel = function (oDeviceInfo, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var newOptions = {};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (xmlData) {
                        var arrXml = [];
                        arrXml.push("<InputProxyChannelList>");
                        var nodeList = NS.$XML(xmlData).find("InputProxyChannel", true);
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            var node = nodeList[i];
                            arrXml.push("<InputProxyChannel>");
                            arrXml.push("<id>" + NS.$XML(node).find("id").eq(0).text() + "</id>");
                            arrXml.push("<name>" + m_utilsInc.escape(NS.$XML(node).find("name").eq(0).text()) + "</name>");
                            arrXml.push("</InputProxyChannel>")
                        }
                        arrXml.push("</InputProxyChannelList>");
                        var xmlDoc = m_utilsInc.loadXML(arrXml.join(""));
                        if (options.success) {
                            options.success(xmlDoc)
                        }
                        resolve(xmlDoc)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.getDigitalChannel, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getDigitalChannelInfo = function (oDeviceInfo, options) {
            let oPromise = new Promise(async (resolve, reject) => {
                var oDigitalChannelXML = null, oDigitalChannelName = {};
                try {
                    oDigitalChannelXML = await m_ISAPIProtocol.getDigitalChannel(oDeviceInfo, {})
                } catch (oError) {
                    reject(oError)
                }
                var nodeList = NS.$XML(oDigitalChannelXML).find("InputProxyChannel", true);
                for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                    var node = nodeList[i], szId = NS.$XML(node).find("id").eq(0).text(),
                        szName = NS.$XML(node).find("name").eq(0).text();
                    oDigitalChannelName[szId] = szName
                }
                var newOptions = {};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (xmlData) {
                        var arrXml = [];
                        arrXml.push("<InputProxyChannelStatusList>");
                        var nodeList = NS.$XML(xmlData).find("InputProxyChannelStatus", true);
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            var node = nodeList[i], szId = NS.$XML(node).find("id").eq(0).text();
                            arrXml.push("<InputProxyChannelStatus>");
                            arrXml.push("<id>" + szId + "</id>");
                            arrXml.push("<sourceInputPortDescriptor>");
                            arrXml.push("<proxyProtocol>" + NS.$XML(node).find("proxyProtocol").eq(0).text() + "</proxyProtocol>");
                            arrXml.push("<addressingFormatType>" + NS.$XML(node).find("addressingFormatType").eq(0).text() + "</addressingFormatType>");
                            arrXml.push("<ipAddress>" + NS.$XML(node).find("ipAddress").eq(0).text() + "</ipAddress>");
                            arrXml.push("<managePortNo>" + NS.$XML(node).find("managePortNo").eq(0).text() + "</managePortNo>");
                            arrXml.push("<srcInputPort>" + NS.$XML(node).find("srcInputPort").eq(0).text() + "</srcInputPort>");
                            arrXml.push("<userName>" + m_utilsInc.escape(NS.$XML(node).find("userName").eq(0).text()) + "</userName>");
                            arrXml.push("<streamType>" + NS.$XML(node).find("streamType").eq(0).text() + "</streamType>");
                            arrXml.push("<online>" + NS.$XML(node).find("online").eq(0).text() + "</online>");
                            arrXml.push("<name>" + m_utilsInc.escape(oDigitalChannelName[szId]) + "</name>");
                            arrXml.push("</sourceInputPortDescriptor>");
                            arrXml.push("</InputProxyChannelStatus>")
                        }
                        arrXml.push("</InputProxyChannelStatusList>");
                        var xmlDoc = m_utilsInc.loadXML(arrXml.join(""));
                        if (options.success) {
                            options.success(xmlDoc)
                        }
                        resolve(xmlDoc)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, m_ISAPIProtocol.CGI.getDigitalChannelInfo, newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.getZeroChannelInfo = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getZeroChannelInfo, options)
        };
        ISAPIProtocol.prototype.getStreamChannels = function (oDeviceInfo, options) {
            if (oDeviceInfo.iAnalogChannelNum != 0) {
                var url = m_utilsInc.formatString(this.CGI.getStreamChannels.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort)
            } else {
                var url = m_utilsInc.formatString(this.CGI.getStreamChannels.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort)
            }
            let szURI;
            if (oDeviceInfo.iAnalogChannelNum != 0) {
                szURI = this.CGI.getStreamChannels.analog
            } else {
                szURI = this.CGI.getStreamChannels.digital
            }
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, szURI, options)
        };
        ISAPIProtocol.prototype.getPPPoEStatus = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getPPPoEStatus, options)
        };
        ISAPIProtocol.prototype.getUPnPPortStatus = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getUPnPPortStatus, options)
        };
        ISAPIProtocol.prototype.getNetworkBond = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getNetworkBond, options)
        };
        ISAPIProtocol.prototype.getNetworkInterface = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getNetworkInterface, options)
        };
        ISAPIProtocol.prototype.getPortInfo = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.getPortInfo, options)
        };
        ISAPIProtocol.prototype.startRealPlay = function (oDeviceInfo, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var iChannelID = options.iChannelID * 100 + options.iStreamType, szUrl = "";
                var szRtspIP = m_utilsInc.delPort(oDeviceInfo.szIP);
                var iRtspPort = oDeviceInfo.iRtspPort;
                if (options.iPort) {
                    iRtspPort = options.iPort
                }
                if (options.bZeroChannel) {
                    szUrl = m_utilsInc.formatString(oDeviceInfo.oProtocolInc.CGI.startRealPlay.zeroChannels, szRtspIP, iRtspPort, iChannelID)
                } else {
                    szUrl = m_utilsInc.formatString(oDeviceInfo.oProtocolInc.CGI.startRealPlay.channels, szRtspIP, iRtspPort, iChannelID)
                }
                var addToWndSet = function () {
                    var wndInfo = new wndInfoClass;
                    wndInfo.iIndex = options.iWndIndex;
                    wndInfo.szIP = oDeviceInfo.szIP;
                    wndInfo.iCGIPort = oDeviceInfo.iCGIPort;
                    wndInfo.szDeviceIdentify = oDeviceInfo.szDeviceIdentify;
                    wndInfo.iChannelID = options.iChannelID;
                    wndInfo.iPlayStatus = PLAY_STATUS_REALPLAY;
                    m_wndSet.push(wndInfo)
                };
                await m_pluginOBJECT.JS_SetSecretKey(0, m_oLocalCfg.secretKey, 1);
                m_pluginOBJECT.JS_Play(szUrl, {
                    auth: oDeviceInfo.szAuth,
                    userInfo: oDeviceInfo.szAuth
                }, options.iWndIndex, "", "", options.bFlag).then(() => {
                    addToWndSet();
                    resolve()
                }, () => {
                    reject()
                })
            });
            return oPromise
        };
        ISAPIProtocol.prototype.startPlay = function (oDeviceInfo, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var addToWndSet = function () {
                    var wndInfo = new wndInfoClass;
                    wndInfo.iIndex = options.iWndIndex;
                    wndInfo.szIP = oDeviceInfo.szIP;
                    wndInfo.szDeviceIdentify = oDeviceInfo.szDeviceIdentify;
                    wndInfo.iPlayStatus = PLAY_STATUS_PLAYBACK;
                    m_wndSet.push(wndInfo)
                };
                m_pluginOBJECT.JS_Play(options.szUrl, {
                    auth: oDeviceInfo.szAuth,
                    userInfo: oDeviceInfo.szAuth
                }, options.iWndIndex, options.startTime, options.endTime, true).then(() => {
                    addToWndSet();
                    resolve()
                }, () => {
                    reject()
                })
            });
            return oPromise
        };
        ISAPIProtocol.prototype.startVoiceTalk = function (oDeviceInfo, iAudioChannel) {
            var szOpenUrl = m_utilsInc.formatString(this.CGI.startVoiceTalk.open, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, iAudioChannel),
                szCloseUrl = m_utilsInc.formatString(this.CGI.startVoiceTalk.close, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, iAudioChannel),
                szAudioDataUrl = m_utilsInc.formatString(this.CGI.startVoiceTalk.audioData, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, iAudioChannel);
            return m_pluginOBJECT.JS_StartTalk(szOpenUrl, szCloseUrl, szAudioDataUrl, oDeviceInfo.szAuth, oDeviceInfo.iAudioType, oDeviceInfo.m_iAudioBitRate, oDeviceInfo.m_iAudioSamplingRate, oDeviceInfo.m_iSoundChan, oDeviceInfo.m_iDeviceAudioSoundChan)
        };
        ISAPIProtocol.prototype.audioPlay = function (options) {
            return m_pluginOBJECT.JS_AudioPlay(options.szUrl, options.szAuth, -1, -1, true, options.iAudioType)
        };
        ISAPIProtocol.prototype.ptzAutoControl = function (oDeviceInfo, bStop, oWndInfo, options) {
            let oPromise = new Promise((resolve, reject) => {
                var iChannelID = oWndInfo.iChannelID, szUrl = "", szData = "";
                options.iPTZSpeed = options.iPTZSpeed < 7 ? options.iPTZSpeed * 15 : 100;
                if (bStop) {
                    options.iPTZSpeed = 0
                }
                if (iChannelID <= oDeviceInfo.iAnalogChannelNum) {
                    szUrl = m_utilsInc.formatString(m_ISAPIProtocol.CGI.ptzAutoControl.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
                } else {
                    szUrl = m_utilsInc.formatString(m_ISAPIProtocol.CGI.ptzAutoControl.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
                }
                szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<autoPanData>" + "<autoPan>" + options.iPTZSpeed + "</autoPan>" + "</autoPanData>";
                var newOptions = {type: "PUT", url: szUrl, data: szData, success: null, error: null};
                var self = this;
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function () {
                        oWndInfo.bPTZAuto = !oWndInfo.bPTZAuto;
                        if (options.success) {
                            options.success()
                        }
                        resolve()
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.ptzControl = function (oDeviceInfo, bStop, oWndInfo, options) {
            var iChannelID = oWndInfo.iChannelID, szUrl = "";
            if (oWndInfo.bPTZAuto) {
                this.ptzAutoControl(oDeviceInfo, true, oWndInfo, {iPTZSpeed: 0})
            }
            if (bStop) {
                options.iPTZSpeed = 0
            } else {
                options.iPTZSpeed = options.iPTZSpeed < 7 ? options.iPTZSpeed * 15 : 100
            }
            var oDirection = [{}, {pan: 0, tilt: options.iPTZSpeed}, {
                pan: 0,
                tilt: -options.iPTZSpeed
            }, {pan: -options.iPTZSpeed, tilt: 0}, {pan: options.iPTZSpeed, tilt: 0}, {
                pan: -options.iPTZSpeed,
                tilt: options.iPTZSpeed
            }, {pan: -options.iPTZSpeed, tilt: -options.iPTZSpeed}, {
                pan: options.iPTZSpeed,
                tilt: options.iPTZSpeed
            }, {
                pan: options.iPTZSpeed,
                tilt: -options.iPTZSpeed
            }, {}, {speed: options.iPTZSpeed}, {speed: -options.iPTZSpeed}, {speed: options.iPTZSpeed}, {speed: -options.iPTZSpeed}, {speed: options.iPTZSpeed}, {speed: -options.iPTZSpeed}];
            var szData = "";
            var oCommond = {};
            switch (options.iPTZIndex) {
                case 1:
                case 2:
                case 3:
                case 4:
                case 5:
                case 6:
                case 7:
                case 8:
                    oCommond = this.CGI.ptzControl;
                    szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<PTZData>" + "<pan>" + oDirection[options.iPTZIndex].pan + "</pan>" + "<tilt>" + oDirection[options.iPTZIndex].tilt + "</tilt>" + "</PTZData>";
                    break;
                case 10:
                case 11:
                    oCommond = this.CGI.ptzControl;
                    szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<PTZData>" + "<zoom>" + oDirection[options.iPTZIndex].speed + "</zoom>" + "</PTZData>";
                    break;
                case 12:
                case 13:
                    oCommond = this.CGI.ptzFocus;
                    szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<FocusData>" + "<focus>" + oDirection[options.iPTZIndex].speed + "</focus>" + "</FocusData>";
                    break;
                case 14:
                case 15:
                    oCommond = this.CGI.ptzIris;
                    szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<IrisData>" + "<iris>" + oDirection[options.iPTZIndex].speed + "</iris>" + "</IrisData>";
                    break;
                default:
                    return
            }
            if (iChannelID <= oDeviceInfo.iAnalogChannelNum) {
                szUrl = m_utilsInc.formatString(oCommond.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
            } else {
                szUrl = m_utilsInc.formatString(oCommond.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
            }
            var newOptions = {type: "PUT", url: szUrl, data: szData, success: null, error: null};
            m_utilsInc.extend(newOptions, options);
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.setPreset = function (oDeviceInfo, oWndInfo, options) {
            var iChannelID = oWndInfo.iChannelID, szUrl = "", szData = "";
            if (iChannelID <= oDeviceInfo.iAnalogChannelNum) {
                szUrl = m_utilsInc.formatString(this.CGI.setPreset.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID, options.iPresetID)
            } else {
                szUrl = m_utilsInc.formatString(this.CGI.setPreset.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID, options.iPresetID)
            }
            szData = "<?xml version='1.0' encoding='UTF-8'?>";
            szData += "<PTZPreset>";
            szData += "<id>" + options.iPresetID + "</id>";
            if (oDeviceInfo.szDeviceType != DEVICE_TYPE_IPDOME) {
                szData += "<presetName>" + "Preset" + options.iPresetID + "</presetName>"
            }
            szData += "</PTZPreset>";
            var newOptions = {type: "PUT", url: szUrl, data: szData, success: null, error: null};
            m_utilsInc.extend(newOptions, options);
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.goPreset = function (oDeviceInfo, oWndInfo, options) {
            var iChannelID = oWndInfo.iChannelID, szUrl = "";
            if (iChannelID <= oDeviceInfo.iAnalogChannelNum) {
                szUrl = m_utilsInc.formatString(this.CGI.goPreset.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID, options.iPresetID)
            } else {
                szUrl = m_utilsInc.formatString(this.CGI.goPreset.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID, options.iPresetID)
            }
            var newOptions = {type: "PUT", url: szUrl, success: null, error: null};
            m_utilsInc.extend(newOptions, options);
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.recordSearch = function (oDeviceInfo, options) {
            let oPromise = new Promise((resolve, reject) => {
                var szUrl = "", szData = "", iChannelID = options.iChannelID, iStreamType = options.iStreamType,
                    szStartTime = options.szStartTime.replace(" ", "T") + "Z",
                    szEndTime = options.szEndTime.replace(" ", "T") + "Z";
                szUrl = m_utilsInc.formatString(m_ISAPIProtocol.CGI.recordSearch, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort);
                szData = "<?xml version='1.0' encoding='UTF-8'?>" + "<CMSearchDescription>" + "<searchID>" + new UUID + "</searchID>" + "<trackList><trackID>" + (iChannelID * 100 + iStreamType) + "</trackID></trackList>" + "<timeSpanList>" + "<timeSpan>" + "<startTime>" + szStartTime + "</startTime>" + "<endTime>" + szEndTime + "</endTime>" + "</timeSpan>" + "</timeSpanList>" + "<maxResults>50</maxResults>" + "<searchResultPostion>" + options.iSearchPos + "</searchResultPostion>" + "<metadataList>" + "<metadataDescriptor>//metadata.ISAPI.org/VideoMotion</metadataDescriptor>" + "</metadataList>" + "</CMSearchDescription>";
                var httpClient = new HttpPluginClient;
                var newOptions = {type: "POST", url: szUrl, data: szData, success: null, error: null};
                m_utilsInc.extend(newOptions, options);
                m_utilsInc.extend(newOptions, {
                    success: function (xmlDoc) {
                        var arrXml = [];
                        arrXml.push("<CMSearchResult>");
                        arrXml.push("<responseStatus>" + NS.$XML(xmlDoc).find("responseStatus").eq(0).text() + "</responseStatus>");
                        arrXml.push("<responseStatusStrg>" + NS.$XML(xmlDoc).find("responseStatusStrg").eq(0).text() + "</responseStatusStrg>");
                        arrXml.push("<numOfMatches>" + NS.$XML(xmlDoc).find("numOfMatches").eq(0).text() + "</numOfMatches>");
                        arrXml.push("<matchList>");
                        var nodeList = NS.$XML(xmlDoc).find("searchMatchItem", true);
                        for (var i = 0, iLen = nodeList.length; i < iLen; i++) {
                            var node = nodeList[i];
                            arrXml.push("<searchMatchItem>");
                            arrXml.push("<trackID>" + NS.$XML(node).find("trackID").eq(0).text() + "</trackID>");
                            arrXml.push("<startTime>" + NS.$XML(node).find("startTime").eq(0).text() + "</startTime>");
                            arrXml.push("<endTime>" + NS.$XML(node).find("endTime").eq(0).text() + "</endTime>");
                            arrXml.push("<playbackURI>" + m_utilsInc.escape(NS.$XML(node).find("playbackURI").eq(0).text()) + "</playbackURI>");
                            arrXml.push("<metadataDescriptor>" + NS.$XML(node).find("metadataDescriptor").eq(0).text().split("/")[1] + "</metadataDescriptor>");
                            arrXml.push("</searchMatchItem>")
                        }
                        arrXml.push("</matchList>");
                        arrXml.push("</CMSearchResult>");
                        xmlDoc = m_utilsInc.loadXML(arrXml.join(""));
                        if (options.success) {
                            options.success(xmlDoc)
                        }
                        resolve(xmlDoc)
                    }, error: function (oError) {
                        if (options.error) {
                            options.error(oError)
                        }
                        reject(oError)
                    }
                });
                m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
            });
            return oPromise
        };
        ISAPIProtocol.prototype.startPlayback = function (oDeviceInfo, options) {
            let oPromise = new Promise(async function (resolve, reject) {
                var iWndIndex = options.iWndIndex, szUrl = "", szStartTime = options.szStartTime,
                    szEndTime = options.szEndTime;
                var szRtspIP = m_utilsInc.delPort(oDeviceInfo.szIP);
                var iRtspPort = oDeviceInfo.iRtspPort;
                if (options.iPort) {
                    iRtspPort = options.iPort
                }
                szUrl = m_utilsInc.formatString(options.cgi, szRtspIP, iRtspPort, options.iChannelID);
                if (!m_utilsInc.isUndefined(options.oTransCodeParam)) {
                    var szTransStreamXml = _generateTransCodeXml(options.oTransCodeParam);
                    if ("" == szTransStreamXml) {
                        return -1
                    }
                    m_pluginOBJECT.JS_SetTrsPlayBackParam(iWndIndex, szTransStreamXml)
                }
                var addToWndSet = function () {
                    var wndInfo = new wndInfoClass;
                    wndInfo.iIndex = iWndIndex;
                    wndInfo.szIP = oDeviceInfo.szIP;
                    wndInfo.iCGIPort = oDeviceInfo.iCGIPort;
                    wndInfo.szDeviceIdentify = oDeviceInfo.szDeviceIdentify;
                    wndInfo.iChannelID = options.iChannelID;
                    wndInfo.iPlayStatus = PLAY_STATUS_PLAYBACK;
                    m_wndSet.push(wndInfo)
                };
                m_pluginOBJECT.JS_Play(szUrl, {
                    auth: oDeviceInfo.szAuth,
                    userInfo: oDeviceInfo.szAuth
                }, iWndIndex, szStartTime, szEndTime, options.bFlag).then(() => {
                    addToWndSet();
                    resolve()
                }, () => {
                    reject()
                })
            });
            return oPromise
        };
        ISAPIProtocol.prototype.reversePlayback = function (oDeviceInfo, options) {
            let oPromise = new Promise(function (resolve, reject) {
                var iWndIndex = options.iWndIndex, szStartTime = options.szStartTime, szEndTime = options.szEndTime;
                var szRtspIP = m_utilsInc.delPort(oDeviceInfo.szIP);
                var iRtspPort = oDeviceInfo.iRtspPort;
                if (options.iPort) {
                    iRtspPort = options.iPort
                }
                var szUrl = m_utilsInc.formatString(options.cgi, szRtspIP, iRtspPort, options.iChannelID);
                m_pluginOBJECT.JS_ReversePlay(szUrl, {
                    auth: oDeviceInfo.szAuth,
                    userInfo: oDeviceInfo.szAuth
                }, iWndIndex, szStartTime, szEndTime).then(() => {
                    var wndInfo = new wndInfoClass;
                    wndInfo.iIndex = iWndIndex;
                    wndInfo.szIP = oDeviceInfo.szIP;
                    wndInfo.iCGIPort = oDeviceInfo.iCGIPort;
                    wndInfo.szDeviceIdentify = oDeviceInfo.szDeviceIdentify;
                    wndInfo.iChannelID = options.iChannelID;
                    wndInfo.iPlayStatus = PLAY_STATUS_REVERSE_PLAYBACK;
                    m_wndSet.push(wndInfo);
                    resolve()
                }, () => {
                    reject()
                })
            });
            return oPromise
        };
        ISAPIProtocol.prototype.startDownloadRecord = function (oDeviceInfo, options) {
            var szUrl = m_utilsInc.formatString(this.CGI.startDownloadRecord, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort);
            var szDownXml = "<?xml version='1.0' encoding='UTF-8'?>" + "<downloadRequest>" + "<playbackURI>" + m_utilsInc.escape(options.szPlaybackURI) + "</playbackURI>" + "</downloadRequest>";
            return m_pluginOBJECT.JS_StartAsyncDownload(szUrl, oDeviceInfo.szAuth, options.szFileName, szDownXml, options.bDateDir)
        };
        ISAPIProtocol.prototype.exportDeviceConfig = function (oDeviceInfo) {
            var szUrl = m_utilsInc.formatString(this.CGI.downloaddeviceConfig, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort);
            return m_pluginOBJECT.JS_DownloadFile(szUrl, oDeviceInfo.szAuth, "", 0)
        };
        ISAPIProtocol.prototype.importDeviceConfig = function (oDeviceInfo, options) {
            var szUrl = m_utilsInc.formatString(this.CGI.uploaddeviceConfig, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort);
            return m_pluginOBJECT.JS_StartAsynUpload(szUrl, "", oDeviceInfo.szAuth, options.szFileName, 0)
        };
        ISAPIProtocol.prototype.restart = function (oDeviceInfo, options) {
            var newOptions = {type: "PUT", success: null, error: null};
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.restart, newOptions)
        };
        ISAPIProtocol.prototype.restore = function (oDeviceInfo, szMode, options) {
            var szUrl = m_utilsInc.formatString(this.CGI.restore, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, szMode);
            var newOptions = {type: "PUT", url: szUrl, success: null, error: null};
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.startUpgrade = function (oDeviceInfo, szFileName) {
            var szUpgradeURL = m_utilsInc.formatString(this.CGI.startUpgrade.upgrade, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort),
                szStatusURL = m_utilsInc.formatString(this.CGI.startUpgrade.status, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort);
            return m_pluginOBJECT.JS_StartUpgrade(szUpgradeURL, "", oDeviceInfo.szAuth, szFileName)
        };
        ISAPIProtocol.prototype.set3DZoom = function (oDeviceInfo, oWndInfo, oPoints, options) {
            var iChannelID = oWndInfo.iChannelID, szUrl = "";
            if (iChannelID <= oDeviceInfo.iAnalogChannelNum) {
                szUrl = m_utilsInc.formatString(this.CGI.set3DZoom.analog, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
            } else {
                szUrl = m_utilsInc.formatString(this.CGI.set3DZoom.digital, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, oWndInfo.iChannelID)
            }
            if (oPoints[0][0] === 0 && oPoints[0][1] === 0 && !(oPoints[2][0] === 0 && oPoints[2][1] === 0)) {
                oPoints[0][0] = oPoints[2][0];
                oPoints[0][1] = oPoints[2][1]
            }
            var szXml = "<?xml version='1.0' encoding='UTF-8'?><Position3D><StartPoint>" + "<positionX>" + parseInt(oPoints[0][0] * 255, 10) + "</positionX>" + "<positionY>" + (255 - parseInt(oPoints[0][1] * 255, 10)) + "</positionY>" + "</StartPoint><EndPoint><positionX>" + parseInt(oPoints[2][0] * 255, 10) + "</positionX>" + "<positionY>" + (255 - parseInt(oPoints[2][1] * 255, 10)) + "</positionY></EndPoint></Position3D>";
            var httpClient = new HttpPluginClient;
            var newOptions = {type: "PUT", url: szUrl, data: szXml, success: null, error: null};
            m_utilsInc.extend(newOptions, options);
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.getSDKCapa = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.SDKCapabilities, options)
        };
        ISAPIProtocol.prototype.deviceCapturePic = function (oDeviceInfo, iChannelID, szPicName, options) {
            var iChannelID = iChannelID * 100 + 1;
            var iRet = -1;
            var szUrl = m_utilsInc.formatString(this.CGI.deviceCapture.channels, oDeviceInfo.szHttpProtocol, oDeviceInfo.szIP, oDeviceInfo.iCGIPort, iChannelID);
            var aQuery = [];
            if (m_utilsInc.isInt(options.iResolutionWidth)) {
                aQuery.push("videoResolutionWidth=" + options.iResolutionWidth)
            }
            if (m_utilsInc.isInt(options.iResolutionHeight)) {
                aQuery.push("videoResolutionHeight=" + options.iResolutionHeight)
            }
            if (aQuery.length > 0) {
                szUrl += "?" + aQuery.join("&")
            }
            var JDeviceCapturePic = function (szUrl, szFileName) {
                var szFileFormat = ".jpg";
                $("body").append('<a id="jsplugin_download_a" href="' + szUrl + '" download=' + szFileName + szFileFormat + '><li id="jsplugin_download_li"></li></a>');
                $("#jsplugin_download_li").trigger("click");
                $("#jsplugin_download_a").remove();
                return 0
            };
            iRet = JDeviceCapturePic(szUrl, szPicName);
            return iRet
        };
        ISAPIProtocol.prototype.digestLogin = function (szIP, iProtocol, iPort, szUserName, szPassword, options) {
            var szHttpProtocol = "";
            if (iProtocol == 2) {
                szHttpProtocol = "https://"
            } else {
                szHttpProtocol = "http://"
            }
            var szUrl = m_utilsInc.formatString(this.CGI.login, szHttpProtocol, szIP, iPort);
            var newOptions = {
                type: "GET",
                url: szUrl,
                auth: m_utilsInc.Base64.encode(":" + szUserName + ":" + szPassword),
                success: null,
                error: null
            };
            var szDeviceIdentify = szIP + "_" + iPort;
            m_utilsInc.extend(newOptions, options);
            return m_webVideoCtrl.I_SendHTTPRequest(szDeviceIdentify, "", newOptions)
        };
        ISAPIProtocol.prototype.getSystemCapa = function (oDeviceInfo, options) {
            return m_webVideoCtrl.I_SendHTTPRequest(oDeviceInfo.szDeviceIdentify, this.CGI.systemCapabilities, options)
        };
        (function (wvc) {
            var XML = function (xd) {
                this.elems = [];
                this.length = 0;
                this.length = this.elems.push(xd)
            };
            XML.prototype.find = function (szNodeName, bRet) {
                var oXmlNode = this.elems[this.length - 1] ? this.elems[this.length - 1].getElementsByTagName(szNodeName) : [];
                this.length = this.elems.push(oXmlNode);
                if (bRet) {
                    return oXmlNode
                } else {
                    return this
                }
            };
            XML.prototype.eq = function (i, bRet) {
                var iLen = this.elems[this.length - 1].length, oXmlNode = null;
                if (iLen > 0 && i < iLen) {
                    oXmlNode = this.elems[this.length - 1][i]
                }
                this.length = this.elems.push(oXmlNode);
                if (bRet) {
                    return oXmlNode
                } else {
                    return this
                }
            };
            XML.prototype.text = function (szText) {
                if (this.elems[this.length - 1]) {
                    if (szText) {
                        if (window.DOMParser) {
                            this.elems[this.length - 1].textContent = szText
                        } else {
                            this.elems[this.length - 1].text = szText
                        }
                    } else {
                        if (window.DOMParser) {
                            return this.elems[this.length - 1].textContent
                        } else {
                            return this.elems[this.length - 1].text
                        }
                    }
                } else {
                    return ""
                }
            };
            XML.prototype.attr = function (szAttrName) {
                if (this.elems[this.length - 1]) {
                    var oAttr = this.elems[this.length - 1].attributes.getNamedItem(szAttrName);
                    if (oAttr) {
                        return oAttr.value
                    } else {
                        return ""
                    }
                }
            };
            wvc.$XML = function (xd) {
                return new XML(xd)
            }
        })(this);
        var Utils = function () {
        };
        Utils.prototype.extend = function () {
            var target = arguments[0] || {}, i = 1, length = arguments.length, options;
            for (; i < length; i++) {
                if ((options = arguments[i]) != null) {
                    for (var name in options) {
                        var src = target[name], copy = options[name];
                        if (target === copy) {
                            continue
                        }
                        if ("object" == typeof copy) {
                            target[name] = this.extend({}, copy)
                        } else if (copy !== undefined) {
                            target[name] = copy
                        }
                    }
                }
            }
            return target
        };
        Utils.prototype.browser = function () {
            var rchrome = /(chrome)[ \/]([\w.]+)/;
            var rsafari = /(safari)[ \/]([\w.]+)/;
            var ropera = /(opera)(?:.*version)?[ \/]([\w.]+)/;
            var rmsie = /(msie) ([\w.]+)/;
            var rmsie2 = /(trident.*rv:)([\w.]+)/;
            var rmozilla = /(mozilla)(?:.*? rv:([\w.]+))?/;
            var ua = navigator.userAgent.toLowerCase();
            var match = rchrome.exec(ua) || rsafari.exec(ua) || ropera.exec(ua) || rmsie.exec(ua) || rmsie2.exec(ua) || ua.indexOf("compatible") < 0 && rmozilla.exec(ua) || ["unknow", "0"];
            if (match.length > 0 && match[1].indexOf("trident") > -1) {
                match[1] = "msie"
            }
            var oBrowser = {};
            oBrowser[match[1]] = true;
            oBrowser.version = match[2];
            return oBrowser
        };
        Utils.prototype.loadXML = function (szXml) {
            if (null == szXml || "" == szXml) {
                return null
            }
            var oXmlDoc = null;
            if (window.DOMParser) {
                var oParser = new DOMParser;
                oXmlDoc = oParser.parseFromString(szXml, "text/xml")
            } else {
                oXmlDoc = new ActiveXObject("Microsoft.XMLDOM");
                oXmlDoc.async = false;
                oXmlDoc.loadXML(szXml)
            }
            return oXmlDoc
        };
        Utils.prototype.toXMLStr = function (oXmlDoc) {
            var szXmlDoc = "";
            try {
                var oSerializer = new XMLSerializer;
                szXmlDoc = oSerializer.serializeToString(oXmlDoc)
            } catch (e) {
                try {
                    szXmlDoc = oXmlDoc.xml
                } catch (e) {
                    return ""
                }
            }
            if (szXmlDoc.indexOf("<?xml") == -1) {
                szXmlDoc = "<?xml version='1.0' encoding='utf-8'?>" + szXmlDoc
            }
            return szXmlDoc
        };
        Utils.prototype.escape = function (szStr) {
            if (szStr) {
                return szStr.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;")
            }
            return szStr
        };
        Utils.prototype.dateFormat = function (oDate, fmt) {
            var o = {
                "M+": oDate.getMonth() + 1,
                "d+": oDate.getDate(),
                "h+": oDate.getHours(),
                "m+": oDate.getMinutes(),
                "s+": oDate.getSeconds(),
                "q+": Math.floor((oDate.getMonth() + 3) / 3),
                S: oDate.getMilliseconds()
            };
            if (/(y+)/.test(fmt)) {
                fmt = fmt.replace(RegExp.$1, (oDate.getFullYear() + "").substr(4 - RegExp.$1.length))
            }
            for (var k in o) {
                if (new RegExp("(" + k + ")").test(fmt)) {
                    fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : ("00" + o[k]).substr(("" + o[k]).length))
                }
            }
            return fmt
        };
        Utils.prototype.Base64 = {
            _keyStr: "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=", encode: function (input) {
                var output = "";
                var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
                var i = 0;
                input = Utils.prototype.Base64._utf8_encode(input);
                while (i < input.length) {
                    chr1 = input.charCodeAt(i++);
                    chr2 = input.charCodeAt(i++);
                    chr3 = input.charCodeAt(i++);
                    enc1 = chr1 >> 2;
                    enc2 = (chr1 & 3) << 4 | chr2 >> 4;
                    enc3 = (chr2 & 15) << 2 | chr3 >> 6;
                    enc4 = chr3 & 63;
                    if (isNaN(chr2)) {
                        enc3 = enc4 = 64
                    } else if (isNaN(chr3)) {
                        enc4 = 64
                    }
                    output = output + this._keyStr.charAt(enc1) + this._keyStr.charAt(enc2) + this._keyStr.charAt(enc3) + this._keyStr.charAt(enc4)
                }
                return output
            }, decode: function (input) {
                var output = "";
                var chr1, chr2, chr3;
                var enc1, enc2, enc3, enc4;
                var i = 0;
                input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
                while (i < input.length) {
                    enc1 = this._keyStr.indexOf(input.charAt(i++));
                    enc2 = this._keyStr.indexOf(input.charAt(i++));
                    enc3 = this._keyStr.indexOf(input.charAt(i++));
                    enc4 = this._keyStr.indexOf(input.charAt(i++));
                    chr1 = enc1 << 2 | enc2 >> 4;
                    chr2 = (enc2 & 15) << 4 | enc3 >> 2;
                    chr3 = (enc3 & 3) << 6 | enc4;
                    output = output + String.fromCharCode(chr1);
                    if (enc3 != 64) {
                        output = output + String.fromCharCode(chr2)
                    }
                    if (enc4 != 64) {
                        output = output + String.fromCharCode(chr3)
                    }
                }
                output = Utils.prototype.Base64._utf8_decode(output);
                return output
            }, _utf8_encode: function (string) {
                string = string.replace(/\r\n/g, "\n");
                var utftext = "";
                for (var n = 0; n < string.length; n++) {
                    var c = string.charCodeAt(n);
                    if (c < 128) {
                        utftext += String.fromCharCode(c)
                    } else if (c > 127 && c < 2048) {
                        utftext += String.fromCharCode(c >> 6 | 192);
                        utftext += String.fromCharCode(c & 63 | 128)
                    } else {
                        utftext += String.fromCharCode(c >> 12 | 224);
                        utftext += String.fromCharCode(c >> 6 & 63 | 128);
                        utftext += String.fromCharCode(c & 63 | 128)
                    }
                }
                return utftext
            }, _utf8_decode: function (utftext) {
                var string = "";
                var i = 0;
                var c = c1 = c2 = 0;
                while (i < utftext.length) {
                    c = utftext.charCodeAt(i);
                    if (c < 128) {
                        string += String.fromCharCode(c);
                        i++
                    } else if (c > 191 && c < 224) {
                        c2 = utftext.charCodeAt(i + 1);
                        string += String.fromCharCode((c & 31) << 6 | c2 & 63);
                        i += 2
                    } else {
                        c2 = utftext.charCodeAt(i + 1);
                        c3 = utftext.charCodeAt(i + 2);
                        string += String.fromCharCode((c & 15) << 12 | (c2 & 63) << 6 | c3 & 63);
                        i += 3
                    }
                }
                return string
            }
        };
        Utils.prototype.createEventScript = function (szFor, szEvent, szHtml) {
            var oScript = document.createElement("script");
            oScript.htmlFor = szFor;
            oScript.event = szEvent;
            oScript.innerHTML = szHtml;
            document.body.parentNode.appendChild(oScript)
        };
        Utils.prototype.isInt = function (str) {
            return /^\d+$/.test(str)
        };
        Utils.prototype.getDirName = function () {
            var szDirName = "";
            if (m_options.szBasePath !== "") {
                szDirName = m_options.szBasePath
            } else {
                var szDirNameRegex = /[^?#]*\//;
                var oScript = document.getElementById("videonode");
                if (oScript) {
                    szDirName = oScript.src.match(szDirNameRegex)[0]
                } else {
                    var aScript = document.scripts;
                    for (var i = 0, iLen = aScript.length; i < iLen; i++) {
                        if (aScript[i].src.indexOf("webVideoCtrl.js") > -1) {
                            oScript = aScript[i];
                            break
                        }
                    }
                    if (oScript) {
                        szDirName = oScript.src.match(szDirNameRegex)[0]
                    }
                }
            }
            return szDirName
        };
        Utils.prototype.loadScript = function (url, callback) {
            var oScript = document.createElement("script");
            oScript.type = "text/javascript";
            oScript.onload = function () {
                callback()
            };
            oScript.src = url;
            document.getElementsByTagName("head")[0].appendChild(oScript)
        };
        Utils.prototype.cookie = function (key, value, options) {
            if (arguments.length > 1 && (value === null || typeof value !== "object")) {
                options = this.extend({}, options);
                if (value === null) {
                    options.expires = -1
                }
                if (typeof options.expires === "number") {
                    var days = options.expires, t = options.expires = new Date;
                    t.setDate(t.getDate() + days)
                }
                return document.cookie = [encodeURIComponent(key), "=", options.raw ? String(value) : encodeURIComponent(String(value)), options.expires ? "; expires=" + options.expires.toUTCString() : "", options.path ? "; path=" + options.path : "; path=/", options.domain ? "; domain=" + options.domain : "", options.secure ? "; secure" : ""].join("")
            }
            options = value || {};
            var result, decode = options.raw ? function (s) {
                return s
            } : decodeURIComponent;
            return (result = new RegExp("(?:^|; )" + encodeURIComponent(key) + "=([^;]*)").exec(document.cookie)) ? decode(result[1]) : null
        };
        Utils.prototype.isUndefined = function (o) {
            return typeof o === "undefined"
        };
        Utils.prototype.isObject = function (o) {
            return Object.prototype.toString.call(o) === "[object Object]"
        };
        Utils.prototype.delPort = function (szIP) {
            var iPos = szIP.indexOf(":");
            if (iPos > -1) {
                return szIP.substring(0, iPos)
            } else {
                return szIP
            }
        };
        Utils.prototype.formatString = function () {
            var string = arguments[0];
            for (var i = 1; i < arguments.length; i++) {
                string = string.replace("%s", arguments[i])
            }
            return string
        };
        Utils.prototype.encodeString = function (str) {
            if (str) {
                return str.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;")
            } else {
                return ""
            }
        };
        Utils.prototype.formatPolygonXmlToJson = function (szXml) {
            var oXml = this.loadXML(szXml);
            var aPolygonList = [];
            var aPoints = [];
            var aRect = [];
            var aAddPolygon = [];
            var aAddRect = [];
            var oData;

            function colorTransfer(szColor) {
                var iValue = parseInt(szColor, 10);
                var szValue = iValue.toString(16);
                szValue = "0" + szValue;
                return szValue.substring(szValue.length - 2)
            }

            $(oXml).find("SnapPolygon").each(function () {
                var iEditType = parseInt($(this).find("EditType, editType").text(), 10) || 0;
                var isClose = $(this).find("isClosed").text() === "true";
                var iPolygonType = parseInt($(this).find("polygonType").text(), 10);
                var fShowSquare = parseFloat($(this).find("showSquare").text()) || 0;
                var szTips = $(this).find("tips").text() || $(this).find("Tips").text();
                var iTipsPos = parseInt($(this).find("tipsPos").text(), 10) || 0;
                var bShowWH = $(this).find("showWH").text() === "true";
                var szColor = "#" + colorTransfer($(this).find("r").text()) + colorTransfer($(this).find("g").text()) + colorTransfer($(this).find("b").text());
                var iMaxPoint = parseInt($(this).find("PointNumMax").text(), 10) - 1;
                var iMinPoint = parseInt($(this).find("MinClosed").text(), 10) - 1;
                var iId = parseInt($(this).find("id").text(), 10);
                var iRedrawMode = parseInt($(this).find("RedrawMode").text(), 10) || 0;
                if ($(this).find("pointList").find("point").length === 0) {
                    if (iPolygonType === 1) {
                        aAddPolygon.push({
                            id: iId,
                            tips: szTips,
                            drawColor: szColor,
                            translucent: .1,
                            maxShapeSupport: 1,
                            maxPointSupport: iMaxPoint,
                            minPointSupport: iMinPoint,
                            showWH: bShowWH,
                            redrawMode: iRedrawMode
                        })
                    } else if (iPolygonType === 0) {
                        aAddRect.push({
                            id: iId,
                            tips: szTips,
                            drawColor: szColor,
                            translucent: .1,
                            widthHeightRate: fShowSquare,
                            maxShapeSupport: 1,
                            type: 1,
                            redrawMode: iRedrawMode,
                            tipsPos: iTipsPos
                        })
                    }
                } else {
                    aPoints = [];
                    $(this).find("pointList").find("point").each(function () {
                        aPoints.push([parseFloat($(this).find("x").text()), parseFloat($(this).find("y").text())])
                    });
                    oData = {
                        id: iId,
                        editType: iEditType,
                        points: aPoints,
                        closed: isClose,
                        tips: szTips,
                        drawColor: szColor,
                        maxPointSupport: iMaxPoint,
                        minPointSupport: iMinPoint,
                        translucent: .1,
                        redrawMode: iRedrawMode
                    };
                    if (iPolygonType === 1) {
                        oData["showWH"] = bShowWH;
                        aPolygonList.push(oData)
                    } else if (iPolygonType === 0) {
                        oData["widthHeightRate"] = fShowSquare;
                        oData["type"] = 1;
                        oData.tipsPos = iTipsPos;
                        aRect.push(oData)
                    }
                }
            });
            return {aRect: aRect, aPolygon: aPolygonList, aAddRect: aAddRect, aAddPolygon: aAddPolygon}
        };
        Utils.prototype.formatPolygonJsonToXml = function (aData) {
            function colorRgb(szHex) {
                var sColor = szHex.toLowerCase();
                var reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/;
                if (sColor && reg.test(sColor)) {
                    var i;
                    if (sColor.length === 4) {
                        var sColorNew = "#";
                        for (i = 1; i < 4; i += 1) {
                            sColorNew += sColor.slice(i, i + 1).concat(sColor.slice(i, i + 1))
                        }
                        sColor = sColorNew
                    }
                    var aColorChange = [];
                    for (i = 1; i < 7; i += 2) {
                        aColorChange.push(parseInt("0x" + sColor.slice(i, i + 2), 16))
                    }
                    return aColorChange
                }
                return [0, 0, 0]
            }

            var aPolygon = aData[0];
            var aRect = aData[1];
            var szXml = "<?xml version='1.0' encoding='utf-8'?><SnapPolygonList>";
            var that = this;
            $.each(aPolygon, function (index, oVal) {
                var aColor = [0, 0, 0];
                if (oVal.drawColor) {
                    aColor = colorRgb(oVal.drawColor)
                } else {
                    aColor = colorRgb("#FF0000")
                }
                szXml += "<SnapPolygon>";
                szXml += "<id>" + oVal.id + "</id>";
                oVal.tips = that.encodeString(oVal.tips);
                if (!oVal.tips) {
                    szXml += "<tips></tips>"
                } else {
                    szXml += "<tips>" + oVal.tips + "</tips>"
                }
                szXml += "<isClosed>" + oVal.closed.toString() + "</isClosed>";
                szXml += "<color><r>" + aColor[0] + "</r><g>" + aColor[1] + "</g><b>" + aColor[2] + "</b></color>";
                szXml += "<polygonType>1</polygonType>";
                szXml += "<PointNumMax>" + (oVal.pointNumMax ? oVal.pointNumMax : 10) + "</PointNumMax>";
                szXml += "<MinClosed>" + (oVal.minClosed ? oVal.minClosed : 4) + "</MinClosed>";
                szXml += "<pointList>";
                $.each(oVal.points, function (i, aVal) {
                    szXml += "<point><x>" + aVal[0] + "</x><y>" + aVal[1] + "</y></point>"
                });
                szXml += "</pointList>";
                szXml += "</SnapPolygon>"
            });
            $.each(aRect, function (index, oVal) {
                var aColor = [0, 0, 0];
                if (oVal.drawColor) {
                    aColor = colorRgb(oVal.drawColor)
                } else {
                    aColor = colorRgb("#FF0000")
                }
                szXml += "<SnapPolygon>";
                szXml += "<id>" + oVal.id + "</id>";
                szXml += "<color><r>" + aColor[0] + "</r><g>" + aColor[1] + "</g><b>" + aColor[2] + "</b></color>";
                szXml += "<polygonType>0</polygonType>";
                oVal.tips = that.encodeString(oVal.tips);
                if (!oVal.tips) {
                    szXml += "<tips></tips>"
                } else {
                    szXml += "<tips>" + oVal.tips + "</tips>"
                }
                if (typeof oVal.closed !== "undefined" && oVal.closed !== null) {
                    szXml += "<isClosed>" + oVal.closed.toString() + "</isClosed>"
                } else {
                    szXml += "<isClosed>true</isClosed>"
                }
                szXml += "<pointList>";
                var aRectTmp = [];
                if (oVal.points.length) {
                    var iMinX = 2;
                    var iMaxX = -1;
                    var iMinY = 2;
                    var iMaxY = -1;
                    $.each(oVal.points, function () {
                        if (iMinX > this[0]) {
                            iMinX = this[0]
                        }
                        if (iMinY > this[1]) {
                            iMinY = this[1]
                        }
                        if (iMaxX < this[0]) {
                            iMaxX = this[0]
                        }
                        if (iMaxY < this[1]) {
                            iMaxY = this[1]
                        }
                    });
                    aRectTmp.push([iMinX, iMinY]);
                    aRectTmp.push([iMaxX, iMinY]);
                    aRectTmp.push([iMaxX, iMaxY]);
                    aRectTmp.push([iMinX, iMaxY])
                }
                $.each(aRectTmp, function (i, aVal) {
                    szXml += "<point><x>" + aVal[0] + "</x><y>" + aVal[1] + "</y></point>"
                });
                szXml += "</pointList>";
                szXml += "</SnapPolygon>"
            });
            szXml += "</SnapPolygonList>";
            return szXml
        };
        Utils.prototype.convertToUTCTime = function (szLocalTime, szFormat) {
            if (typeof szFormat == "undefined") {
                szFormat = "yyyy-MM-dd hh:mm:ss"
            }
            szLocalTime = szLocalTime.replace("T", " ").replace("Z", "");
            var _dLocalDate = new Date(Date.parse(szLocalTime.replace(/-/g, "/")));
            _dLocalDate = this.utcDateFormat(_dLocalDate, szFormat);
            _dLocalDate = _dLocalDate.replace(" ", "T");
            return _dLocalDate
        };
        Utils.prototype.utcDateFormat = function (oDate, fmt) {
            var o = {
                "M+": oDate.getUTCMonth() + 1,
                "d+": oDate.getUTCDate(),
                "h+": oDate.getUTCHours(),
                "m+": oDate.getUTCMinutes(),
                "s+": oDate.getUTCSeconds(),
                "q+": Math.floor((oDate.getUTCMonth() + 3) / 3),
                S: oDate.getUTCMilliseconds()
            };
            if (/(y+)/.test(fmt)) {
                fmt = fmt.replace(RegExp.$1, (oDate.getUTCFullYear() + "").substr(4 - RegExp.$1.length))
            }
            for (var k in o) {
                if (new RegExp("(" + k + ")").test(fmt)) {
                    fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : ("00" + o[k]).substr(("" + o[k]).length))
                }
            }
            return fmt
        };
        Utils.prototype.convertToLocalTime = function (szUTCTime, iDiffTime) {
            szUTCTime = szUTCTime.replace("T", " ").replace("Z", "");
            if (typeof iDiffTime == "undefined") {
                iDiffTime = 0
            }
            var szFormat = "yyyy-MM-dd hh:mm:ss";
            var _aDate = szUTCTime.split(" ")[0].split("-");
            var _iFullYear = parseInt(_aDate[0], 10);
            var _iMonth = parseInt(_aDate[1], 10) - 1;
            var _iDay = parseInt(_aDate[2], 10);
            var _aTimes = szUTCTime.split(" ")[1].split(":");
            var _iHour = parseInt(_aTimes[0], 10);
            var _iMinute = parseInt(_aTimes[1], 10);
            var _iSecond = parseInt(_aTimes[2], 10);
            var _dLocalDate = new Date(Date.UTC(_iFullYear, _iMonth, _iDay, _iHour, _iMinute, _iSecond));
            _dLocalDate.setTime(_dLocalDate.getTime() + iDiffTime);
            return this.dateFormat(_dLocalDate, szFormat).replace(" ", "T") + "Z"
        };

        function UUID() {
            this.id = this.createUUID()
        }

        UUID.prototype.valueOf = function () {
            return this.id
        };
        UUID.prototype.toString = function () {
            return this.id
        };
        UUID.prototype.createUUID = function () {
            var dg = new Date(1582, 10, 15, 0, 0, 0, 0);
            var dc = new Date;
            var t = dc.getTime() - dg.getTime();
            var h = "-";
            var tl = UUID.getIntegerBits(t, 0, 31);
            var tm = UUID.getIntegerBits(t, 32, 47);
            var thv = UUID.getIntegerBits(t, 48, 59) + "1";
            var csar = UUID.getIntegerBits(UUID.rand(4095), 0, 7);
            var csl = UUID.getIntegerBits(UUID.rand(4095), 0, 7);
            var n = UUID.getIntegerBits(UUID.rand(8191), 0, 7) + UUID.getIntegerBits(UUID.rand(8191), 8, 15) + UUID.getIntegerBits(UUID.rand(8191), 0, 7) + UUID.getIntegerBits(UUID.rand(8191), 8, 15) + UUID.getIntegerBits(UUID.rand(8191), 0, 15);
            return tl + h + tm + h + thv + h + csar + csl + h + n
        };
        UUID.getIntegerBits = function (val, start, end) {
            var base16 = UUID.returnBase(val, 16);
            var quadArray = new Array;
            var quadString = "";
            var i = 0;
            for (i = 0; i < base16.length; i++) {
                quadArray.push(base16.substring(i, i + 1))
            }
            for (i = Math.floor(start / 4); i <= Math.floor(end / 4); i++) {
                if (!quadArray[i] || quadArray[i] == "") quadString += "0"; else quadString += quadArray[i]
            }
            return quadString
        };
        UUID.returnBase = function (number, base) {
            var convert = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z"];
            if (number < base) var output = convert[number]; else {
                var MSD = "" + Math.floor(number / base);
                var LSD = number - MSD * base;
                if (MSD >= base) var output = this.returnBase(MSD, base) + convert[LSD]; else var output = convert[MSD] + convert[LSD]
            }
            return output
        };
        UUID.rand = function (max) {
            return Math.floor(Math.random() * max)
        };
        m_ISAPIProtocol = new ISAPIProtocol;
        m_utilsInc = new Utils;
        return this
    }();
    var NS = window.WebVideoCtrl = WebVideoCtrl;
    NS.version = "3.3.0"
    console.log("window.WebVideoCtrl:"+window.WebVideoCtrl);
})(this);
if ("object" === typeof exports && typeof module !== "undefined") {
} else if ("function" === typeof define && define.amd) {
    define(function () {
        return WebVideoCtrl
    })
} else if ("function" === typeof define && define.cmd) {
    define(function (require, exports, module) {
        module.exports = WebVideoCtrl
    })
} else {
}